home *** CD-ROM | disk | FTP | other *** search
/ Amiga Plus Special 18 / AMIGAplus Sonderheft 18 (1999)(ICP)(DE)[!].iso / PD / Spiele / InvasionForce / Source / game_play1.c < prev    next >
C/C++ Source or Header  |  1999-01-08  |  135KB  |  3,570 lines

  1. /*
  2.    game_play1.c -- game play module for Invasion Force
  3.  
  4.    This module handles game play: loading and saving games, getting orders
  5.    from the user, moving units, handling combat, and presenting various
  6.    menus for branching to the status reports, message system, etc.
  7.  
  8.    This source code is free.  You may make as many copies as you like.
  9. */
  10.  
  11. #include "global.h"
  12. #include <proto/layers.h>
  13.  
  14. #define CLEAR_WINDOW() SetRast(rast_port,0);zero_scrollers();display=0;
  15.  
  16. #define EXIT_GAME 1     // kluge values for flow control
  17. #define GAME_OVER 2
  18. #define UNIT_DONE 3     // I mainly use these to break out of nested functions
  19. #define UNIT_LOST 4
  20. #define GO_SURVEY 5
  21. #define EXIT_SURVEY   6
  22. #define GO_MOVEMENT   7
  23. #define END_TURN      8
  24. #define GAME_RESTORED 9
  25. #define GO_PRODUCTION 10
  26. #define CE_RESTART -17
  27.  
  28. int control_flag = 0;   // kluge variable to break out of nested funcs
  29.  
  30. #define INVISIBLE 0
  31. #define VISIBLE 1       // flag values, purely for more readable code
  32.  
  33. char game_filepath[108], game_filename[108];
  34.  
  35. struct Menu *move_menu_strip = NULL;
  36. struct Menu *vey_menu_strip = NULL;
  37. struct Menu *prod_menu_strip = NULL;
  38.  
  39. int PathCost=0;
  40.  
  41. struct PLayer roster[9];
  42. int player = 0;    // number of the current player
  43. int turn = 0;      // current turn number
  44. BOOL display;      // current status of player display
  45. int cursx, cursy;  // current location of the survey cursor
  46. char id_filetag[5];  // identifies temporary files
  47. struct MinList unit_list;   // master list of active game pieces
  48. struct BattleRecord battle;
  49.  
  50. char *prefix = "T:EMP2.";   // prefix for all Invasion force temporary files
  51.  
  52. // set_display_offsets() will set the xoffs and yoffs values to try and
  53. // center the specified col & row map position as closely as possible
  54.  
  55. void set_display_offsets(col,row)
  56. {
  57.    int overlap=(wrap?WRAP_OVERLAP:0);
  58.    int wd=width+2, ht=height+2;
  59.  
  60.    xoffs = col-(disp_wd >> 1);
  61.    if (xoffs<(0-overlap))
  62.       xoffs = (0-overlap);
  63.    if (xoffs>(wd-disp_wd+overlap))
  64.       xoffs = (wd-disp_wd+overlap);
  65.  
  66.    yoffs = row-(disp_ht >> 1);
  67.    if (yoffs<-overlap)
  68.       yoffs = -overlap;
  69.    if (yoffs>ht-disp_ht+overlap)
  70.       yoffs = ht-disp_ht+overlap;
  71.  
  72.    update_scrollers();
  73. }
  74.  
  75.  
  76. /*
  77.    count_units_at() -- return number of units at the specified map sector.
  78.    The AO can call this function to see how many units he has on a specified
  79.    sector, but should only call it for sectors he controls.  Using it to see
  80.    how many units the enemy has in a sector would be cheating!
  81. */
  82.  
  83. int count_units_at(col,row)
  84. int col,row;
  85. {
  86.    int ctr = 0;
  87.    struct Unit *unit = (struct Unit *)unit_list.mlh_Head;
  88.  
  89.    for (; unit->unode.mln_Succ; unit = (struct Unit *)unit->unode.mln_Succ)
  90.       if (unit->col==col && unit->row==row)
  91.          ctr++;
  92.    return ctr;
  93. }
  94.  
  95.  
  96. // this is the case where the player has just seen a city, so we
  97. // must create an icon for it on his map
  98.  
  99. void add_city_to_player_map(player,metro)
  100. int player;
  101. struct City *metro;
  102. {
  103.    struct MapIcon *icon = AllocVec((long)sizeof(*icon),MEMF_CLEAR);
  104.  
  105.    if (icon==NULL) return;    // Will this mean Phantom Cities?
  106.  
  107.    // copy basic information
  108.    icon->col = metro->col;
  109.    icon->row = metro->row;
  110.    icon->type = CITY;
  111.    icon->owner = metro->owner;
  112.  
  113.    if (metro->owner==player)
  114.       icon->stacked = (count_units_at(icon->col,icon->row)>1);
  115.    else
  116.       icon->stacked = FALSE;
  117.    AddTail((struct List *)&PLAYER.icons,(struct Node *)icon);
  118. }
  119.  
  120.  
  121. // similar to function above
  122. // adds an icon to the map, this time for a unit
  123.  
  124. void add_icon_to_player_map(player,unit)
  125. int player;
  126. struct Unit *unit;
  127. {
  128.    struct MapIcon *icon = AllocVec((long)sizeof(*icon),MEMF_CLEAR);
  129.  
  130.    if (icon==NULL) return;
  131.  
  132.    // copy basic information
  133.    icon->col = unit->col;
  134.    icon->row = unit->row;
  135.    icon->type = unit->type;
  136.    icon->owner = unit->owner;
  137.    icon->token = ORDER_NONE;
  138.    if (unit->owner==player && unit->orders!=NULL)
  139.       icon->token = unit->orders->type;
  140.  
  141.    if (player==unit->owner)
  142.       icon->stacked = (count_units_at(icon->col,icon->row)>1);
  143.    else
  144.       icon->stacked = FALSE;
  145.    AddTail((struct List *)&PLAYER.icons,(struct Node *)icon);
  146. }
  147.  
  148.  
  149. /*
  150.    submarines cannot see land or air units
  151.    when a hex is being explored, this function determines whether
  152.    a land or air unit is seen either by a sub or by some other adjacent
  153.    unit or city; returns TRUE if seen, FALSE otherwise
  154. */
  155. BOOL seenby_subP(player,targ)
  156. int player;
  157. struct Unit *targ;
  158. {
  159.    int col=targ->col, row=targ->row;
  160.    int num_hexes=adjacent(col,row);
  161.    struct City *metro;
  162.    struct Unit *unit;
  163.    int ctr;
  164.  
  165.    if (targ->owner==player)
  166.       return TRUE;
  167.  
  168.    if (wishbook[targ->type].ship_flag)  // subs can always see ships
  169.       return TRUE;
  170.  
  171.    // search adjacent hexes for cities or units that can spot the target
  172.    for (ctr=0; ctr<num_hexes; ctr++) {
  173.       col = hexlist[ctr].col;
  174.       row = hexlist[ctr].row;
  175.       if (metro = city_hereP(col,row))
  176.          if (metro->owner==player)
  177.             return TRUE;
  178.       for (unit=(struct Unit *)unit_list.mlh_Head;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  179.          if (unit->col==col && unit->row==row && unit->owner==player)
  180.             if (unit->type!=SUB)
  181.                return TRUE;
  182.    OD
  183.    return FALSE;  // nobody spotted him
  184. }
  185.  
  186.  
  187. // determine whether a submarine at the given position is visible
  188. // returns TRUE if the unit "sub" can be seen by any cities or other
  189. // units owned by "player", otherwise FALSE
  190.  
  191. BOOL sub_seenP(player,sub)
  192. int player;    // the player who is exploring
  193. struct Unit *sub;
  194. {
  195.    int col=sub->col, row=sub->row;
  196.    int terrain=get(t_grid,col,row);
  197.    int num_hexes=adjacent(col,row);
  198.    struct City *metro;
  199.    struct Unit *unit;
  200.    int ctr;
  201.  
  202.    if (sub->owner==player)
  203.       return TRUE;
  204.  
  205.    if (terrain==HEX_SHALLOWS)   // subs always visible in shallow water
  206.       return TRUE;
  207.  
  208.    // search adjacent hexes for cities or units that can spot the sub
  209.    for (ctr=0; ctr<num_hexes; ctr++) {
  210.       col = hexlist[ctr].col;
  211.       row = hexlist[ctr].row;
  212.       if (metro = city_hereP(col,row))
  213.          if (metro->owner==player)
  214.             return TRUE;
  215.       for (unit=(struct Unit *)unit_list.mlh_Head;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  216.          if (unit->col==col && unit->row==row && unit->owner==player)
  217.             if (unit->type==CRUISER || unit->type==DESTROYER || unit->type==SUB)
  218.                return TRUE;
  219.    OD
  220.    return FALSE;  // nobody spotted him
  221. }
  222.  
  223.  
  224. /*
  225.    recon() determines if a given city is vulnerable to recon by the current
  226.    player's aircraft (in other words, if the player has a plane in an
  227.    adjacent hex to see what the city is producing); if the city is
  228.    vulnerable to recon, we will store the production type in city.recon[]
  229. */
  230.  
  231. void recon(metro)
  232. struct City *metro;
  233. {
  234.    int hexes, ctr;
  235.  
  236.    if (metro==NULL)
  237.       return;
  238.    if (metro->owner==player || metro->owner==0)
  239.       return;
  240.  
  241.    hexes = adjacent(metro->col,metro->row);   // build list of adjacent hexes
  242.    for (ctr=0;ctr<hexes;ctr++) {
  243.       struct Unit *unit = (struct Unit *)unit_list.mlh_Head;
  244.  
  245.       // search this hex for recon aircraft (currently only fighters)
  246.       for (;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  247.          if (unit->col==hexlist[ctr].col && unit->row==hexlist[ctr].row)
  248.             if (unit->owner==player && unit->type==FIGHTER) {
  249.                metro->recon[player] = metro->unit_type;
  250.                return;
  251.             FI
  252.    }
  253. }
  254.  
  255.  
  256. // very important -- function to explore a hex for a certain player
  257.  
  258. void explore_hex(player,col,row,visible,forced)
  259. int player, col, row, visible, forced;
  260. { // explore an individual hex
  261.  
  262.    // try wrapping the value around onto the map
  263.    wrap_coords(&col,&row);
  264.  
  265.    // now see what terrain is here
  266.    put(PLAYER.map,col,row,get(t_grid,col,row));
  267.  
  268.    // also roads
  269.    if (get_flags(t_grid,col,row)&ROAD)
  270.       put_flags(PLAYER.map,col,row,get_flags(PLAYER.map,col,row)|ROAD);
  271.    else
  272.       put_flags(PLAYER.map,col,row,get_flags(PLAYER.map,col,row)&(~ROAD));
  273.  
  274.    /* clear any outdated icons from the player's map, this position */
  275.    {
  276.       BOOL zapped;
  277.  
  278.       do {
  279.          struct MapIcon *icon = (struct MapIcon *)PLAYER.icons.mlh_Head;
  280.  
  281.          zapped = FALSE;
  282.          for (; icon->inode.mln_Succ; icon = (struct MapIcon *)icon->inode.mln_Succ) {
  283.             if (icon->col==col && icon->row==row) {
  284.                Remove((struct Node *)icon);
  285.                FreeVec(icon);
  286.                zapped = TRUE;
  287.                break;
  288.             FI
  289.          OD
  290.       } while (zapped);
  291.    }
  292.  
  293.    // update any city that might be here
  294.    {
  295.       struct City *metro = (struct City *)city_list.mlh_Head;
  296.       for ( ; metro->cnode.mln_Succ; metro = (struct City *)metro->cnode.mln_Succ)
  297.          if (metro->col==col && metro->row==row) {
  298.             add_city_to_player_map(player,metro);
  299.             recon(metro);   // try to recon the city
  300.             goto eh_final_update;  /* don't display military units if a city is here */
  301.          FI
  302.    }
  303.  
  304.    // finally, search for new up-to-date pieces and map them
  305.    {
  306.       struct Unit *unit = (struct Unit *)unit_list.mlh_Head;
  307.       BOOL mapped=FALSE;
  308.  
  309.       for (; unit->unode.mln_Succ; unit = (struct Unit *)unit->unode.mln_Succ)
  310.          if (unit->row==row && unit->col==col) { // found one!
  311.             if (unit->owner!=player)
  312.                clear_orders(unit);  // enemy units come off sentry now
  313.             if (unit->ship)  continue;    /* don't map units on board ships */
  314.             if (unit->type==SUB)    // subs sometimes not visible, of course
  315.                if (sub_seenP(player,unit)==FALSE)
  316.                   continue;
  317.             if (seenby_subP(player,unit)==FALSE && forced==FALSE)
  318.                continue;
  319.             if (mapped==FALSE) {
  320.                mapped = TRUE;  // only map the first one
  321.                add_icon_to_player_map(player,unit);
  322.             FI
  323.          FI
  324.    }
  325.  
  326.  eh_final_update:
  327.    if (visible)  //NOTE: this is now a BOOLEAN value!
  328.       GP_update_hex_display(col,row);
  329. }
  330.  
  331.  
  332. /*
  333.    given a hex, explore_at_hex() will explore it and everyplace
  334.    directly adjacent -- this is the normal way that exploration happens
  335.    in Invasion Force
  336. */
  337. void explore_at_hex(player,col,row,visible,forced)
  338. int player, col, row, visible, forced;
  339. { // explore everything in and adjacent to a given hex
  340.    explore_hex(player,col,row,visible,forced);
  341.    explore_hex(player,col-1,row,visible,forced);
  342.    explore_hex(player,col+1,row,visible,forced);
  343.    explore_hex(player,col,row-1,visible,forced);
  344.    explore_hex(player,col,row+1,visible,forced);
  345.    if (row%2) {  /* i.e. if it's an odd-numbered column */
  346.       explore_hex(player,col+1,row-1,visible,forced);
  347.       explore_hex(player,col+1,row+1,visible,forced);
  348.    } else {
  349.       explore_hex(player,col-1,row-1,visible,forced);
  350.       explore_hex(player,col-1,row+1,visible,forced);
  351.    FI
  352. }
  353.  
  354.  
  355. // gives the city to a player, either his home city at the start of the game,
  356. // or else a city he has conquered with a military unit -- and includes a
  357. // call to examine_city() in the "status.c" module
  358.  
  359. void conquer_city(taken_city)
  360. struct City *taken_city;
  361. {  // take a city
  362.    int prior_owner=taken_city->owner;
  363.    BOOL vis=Bool(PLAYER.show&SHOW_GRP);  // visibility
  364.  
  365.    /*
  366.       When this function is called, the map should *already* be displayed
  367.       and scrolled to the proper location, so the city is positioned on the
  368.       screen; conquer_city() does NOT do this automatically.
  369.    */
  370.    taken_city->owner = player;   // city mine now!
  371.    // update the map icons for both old and new owners
  372.    if (prior_owner!=0)     // no prior owner of neutral cities
  373.       explore_hex(prior_owner,taken_city->col,taken_city->row,INVISIBLE,TRUE);
  374.    explore_at_hex(player,taken_city->col,taken_city->row,vis,TRUE);
  375.    if (vis) {
  376.       // do this so the user can see which city he is looking at
  377.       save_hex_graphics(taken_city->col,taken_city->row,0);    // blit the background to a safe place
  378.       plot_mapobject(taken_city->col,taken_city->row,MAP_MARKER);
  379.    FI
  380.    // set production to -1 to show that we will be restarting
  381.    // production from zilch
  382.    taken_city->unit_type = -1;
  383.    taken_city->unit_wip = 0;
  384.    if (ISHUMAN(PLAYER.type)) {
  385.       // create requester -- let him select production
  386.       examine_city(taken_city);
  387.       // unmark the city on screen
  388.       restore_hex_graphics(taken_city->col,taken_city->row,0);
  389.    FI
  390.    if (NONHUMAN(PLAYER.type)) {
  391.       // we call the computer player version of selecting production
  392.       set_automated_production(taken_city);
  393.    FI
  394. }
  395.  
  396.  
  397. // return the player who controls the specified hex, or 0 if nobody does
  398. int hex_owner(col,row)
  399. int col,row;
  400. {
  401.    struct City *metro;
  402.    struct Unit *unit=(struct Unit *)unit_list.mlh_Head;
  403.  
  404.    // search the cities
  405.    if (metro = city_hereP(col,row))
  406.       return (int)metro->owner;
  407.  
  408.    // search the military units
  409.    for (; unit->unode.mln_Succ; unit=(struct Unit *)unit->unode.mln_Succ)
  410.       if (unit->col==col && unit->row==row)
  411.          return (int)unit->owner;
  412.  
  413.    return 0;
  414. }
  415.  
  416.  
  417. // quick macro to see if a unit has been killed
  418. #define ALIVE(a) (a->damage<wishbook[a->type].hitpoints)
  419.  
  420.  
  421. /***
  422.    choose_defender() == select a unit to defend a hex that has come under attack
  423.    This is supposed to handle the complexities of combat that were introduced
  424.    when I implemented stacking rules, particularly when different kinds of
  425.    units are stacked in the hex that is under attack
  426. ***/
  427.  
  428. struct Unit *choose_defender(attacker,targx,targy)
  429. struct Unit *attacker;
  430. int targx, targy;
  431. {
  432.    struct Unit *unit, *defender=NULL;
  433.    static int aapt[] = { 4, 5, 3,  2,  1, 6, 7, 8, 10, 11, 9, 99, 99 };
  434.    static int sapt[] = { 1, 2, 3, 11, 10, 4, 5, 6,  8,  9, 7, 99, 99 };
  435.    int *table = sapt;
  436.    int priority = 99;
  437.    int terrain = get(t_grid,targx,targy);
  438.  
  439.    // first determine if I should use the AAPT or the SAPT
  440.    if (attacker->type==FIGHTER)
  441.       table = aapt;
  442.  
  443.    for (unit=(struct Unit *)unit_list.mlh_Head;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  444.       if (unit->col==targx && unit->row==targy)
  445.          if (table[unit->type]<priority) {
  446.             // I have to throw out any ground units that are on board a ship
  447.             if (OCEAN_P(terrain) && (unit->type==RIFLE || unit->type==ARMOR))
  448.                continue;
  449.             // also throw out any fighters that are landed (not overflying)
  450.             if (unit->ship)
  451.                continue;
  452.             defender = unit;
  453.             priority = table[unit->type];
  454.          FI
  455.  
  456.    if (defender==NULL)
  457.       print("ERROR: choose_defender() failed to find defending unit for combat!\n");
  458.  
  459.    return defender;
  460. }
  461.  
  462.  
  463. /***
  464.    attack_hex() == universal attack function
  465.    All attacks in the game are targetted against a hex, not against specific
  466.    enemy units.  This function calculates the defensive strength of the target
  467.    hex and resolves the conflict.  The outcome is stored in "control_flag" and
  468.    in the combat report.
  469. ***/
  470.  
  471. void attack_hex(attacker,targx,targy)
  472. struct Unit *attacker;
  473. int targx, targy;
  474. {
  475.    int att_player=attacker->owner, def_player=0;// attacking and defending players
  476.    int def_terrain;
  477.    int attfact, deffact, movement_cost;
  478.    struct Unit *defender=NULL;
  479.    struct Unit militia;
  480.    BOOL bombardment=FALSE;
  481.    BOOL white_flag=FALSE;
  482.    ULONG num_blows, bit_blows;
  483.    struct City *captured=NULL;
  484.  
  485.    clear_orders(attacker);    // just to make sure
  486.  
  487.    /*
  488.       First part of this function is devoted to detecting various kinds of
  489.       invalid attacks and weeding them out before we get to the good stuff.
  490.    */
  491.  
  492.    // has the unit already attacked something this turn?
  493.    // fighters can make multiple attacks per turn; other cannot
  494. // I have disabled this until I get a beter idea how it should work
  495. //   if (attacker->type!=FIGHTER && attacker->attacks>0) {
  496. //      tell_user2("That unit has already attacked once this turn.",FALSE,DONK_SOUND);
  497. //      return;     // other units cannot
  498. //   FI
  499.  
  500.    if (city_hereP(targx,targy))
  501.       switch (attacker->type) {
  502.          case BOMBER:   // bomber can attack city, but not capture it
  503.             bombardment = TRUE;
  504.          case ARMOR:
  505.          case RIFLE:
  506.          case AIRCAV:
  507.             break;      // these units can attack cities
  508.          default:       // other units cannot
  509.             if (PLAYER.show&SHOW_REQ) {
  510.                playSound(DONK_SOUND,PLAYER.snd_vol);
  511.                alert(map_window,"Information...","That unit cannot attack a city.","Okay");
  512.             }
  513.             return;
  514.       };
  515.  
  516.    // bombers and aircav cannot attack any hex containing airborne craft
  517.    // aircraft on carriers or in cities or airfields are not considered airborne
  518.    if (attacker->type==BOMBER || attacker->type==AIRCAV)
  519.       if (city_hereP(targx,targy)==NULL) {
  520.          struct Unit *unit = (struct Unit *)unit_list.mlh_Head;
  521.          for (;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  522.             if (unit->ship==NULL && unit->col==targx && unit->row==targy)
  523.                switch (unit->type) {
  524.                   case FIGHTER:
  525.                   case BOMBER:
  526.                   case AIRCAV:
  527.                      if (PLAYER.show&SHOW_REQ) {
  528.                         playSound(DONK_SOUND,PLAYER.snd_vol);
  529.                         alert(map_window,"Information...","Bombers or air cav cannot attack other aircraft.","Okay");
  530.                      }
  531.                      return;
  532.                }
  533.       FI
  534.  
  535.    def_terrain = get(t_grid,targx,targy);
  536.    if (city_hereP(targx,targy)) {
  537.       def_terrain = HEX_CITY;
  538.       // create a dummy "militia unit" to defend the city
  539.       militia.col = targx;
  540.       militia.row = targy;
  541.       militia.owner = city_hereP(targx,targy)->owner;
  542.       militia.type = RIFLE;
  543.       militia.damage = 0;
  544.       militia.orders = NULL;
  545.       defender = &militia;
  546.       def_player = city_hereP(targx,targy)->owner;
  547.       // set flag to show we are using a militia unit
  548.       white_flag = TRUE;
  549.    };
  550.  
  551.    /* find the defending UNIT in the hex -- the unit we are using to */
  552.    /* determine the defender's strength, etc. */
  553.    if (defender==NULL)
  554.       defender = choose_defender(attacker,targx,targy);
  555.  
  556.    if (defender) {
  557.       clear_orders(defender);    // just to make sure
  558.       if( def_player == 0 )
  559.           def_player = defender->owner;
  560.    } else {
  561.       print("Error finding defender for combat!\n");
  562.       return;
  563.    FI
  564.  
  565.    // when a fighter attacks a hex containing both aircraft and other units,
  566.    // I set the bombardment flag to show the fighter should not enter the
  567.    // hex, even if victorious
  568.    if (attacker->type==FIGHTER) {
  569.       BOOL air=FALSE, surface=FALSE;
  570.       struct Unit *unit;
  571.       for (unit=(struct Unit *)unit_list.mlh_Head;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  572.          if (unit->col==targx && unit->row==targy) {
  573.             if (wishbook[unit->type].range>0)   // aircraft
  574.                air=TRUE;
  575.             else
  576.                surface=TRUE;
  577.             if (air==TRUE && surface==TRUE) {
  578.                bombardment = TRUE;
  579.                break;
  580.             FI
  581.          FI
  582.    FI
  583.  
  584.    // most ships cannot attack ground forces; only cruisers and battleships can
  585.    if (movement_cost_table[attacker->type][def_terrain]<0)
  586.       switch (attacker->type) {
  587.          case TRANSPORT:
  588.          case SUB:
  589.          case DESTROYER:
  590.          case CARRIER:
  591.             if (PLAYER.show&SHOW_REQ) {
  592.                playSound(DONK_SOUND,PLAYER.snd_vol);
  593.                sprintf(foo,"The %s cannot bombard land targets.",
  594.                   wishbook[attacker->type].name);
  595.                (void)rtEZRequestTags(foo,"Oops!",NULL,NULL,
  596.                   RT_Window,        map_window,
  597.                   RT_ReqPos,        REQPOS_CENTERSCR,
  598.                   RT_LockWindow,    TRUE,
  599.                   RTEZ_Flags,       EZREQF_CENTERTEXT,
  600.                   TAG_DONE );
  601.                }
  602.             return;
  603.          case CRUISER:
  604.          case BATTLESHIP:
  605.             bombardment = TRUE;
  606.       }
  607.  
  608.    // riflemen or armor can bombard ships unless aircraft are protecting them
  609.    if (OCEAN_P(def_terrain))
  610.       if (attacker->type==RIFLE || attacker->type==ARMOR) {
  611.          /*
  612.             At this point we might have ground forces trying to attack something
  613.             in a HEX_SHALLOWS terrain with a bridge over it.
  614.  
  615.             Otherwise the ground forces must be trying to bombard an ocean hex.
  616.             This would mean it's attacking a hex with ships or aircraft or both.
  617.          */
  618.          BOOL ship_here=FALSE, aircraft_here=FALSE, bridge_here=FALSE;
  619.          struct Unit *sunit;
  620.  
  621.          // check for roads first
  622.          if (get_flags(t_grid,attacker->col,attacker->row)& \
  623.          get_flags(t_grid,defender->col,defender->row)&ROAD)
  624.             bridge_here = TRUE;
  625.  
  626.          for (sunit=(struct Unit *)unit_list.mlh_Head;sunit->unode.mln_Succ;sunit=(struct Unit *)sunit->unode.mln_Succ)
  627.             if (wishbook[sunit->type].ship_flag && sunit->col==targx && sunit->row==targy) {
  628.                ship_here = TRUE;
  629.                break;
  630.             FI
  631.          for (sunit=(struct Unit *)unit_list.mlh_Head;sunit->unode.mln_Succ;sunit=(struct Unit *)sunit->unode.mln_Succ)
  632.             if (wishbook[sunit->type].range>0 && sunit->col==targx && sunit->row==targy) {
  633.                aircraft_here = TRUE;
  634.                break;
  635.             FI
  636.          /*
  637.             There are three possibilities for the target ocean hex.  It could have
  638.             ships, aircraft, or both ships and aircraft.  Each calls for a slightly
  639.             different response.
  640.          */
  641.          if (aircraft_here && ship_here) {
  642.             if (PLAYER.show & SHOW_REQ) {
  643.                 playSound(DONK_SOUND,PLAYER.snd_vol);
  644.                 (void)rtEZRequestTags(\
  645.                    "Enemy air cover prevents your ground forces\nfrom bombarding ships there.",
  646.                    "Drat!",NULL,NULL,
  647.                    RT_Window,        map_window,
  648.                    RT_ReqPos,        REQPOS_CENTERSCR,
  649.                    RT_LockWindow,    TRUE,
  650.                    RTEZ_Flags,       EZREQF_CENTERTEXT,
  651.                    TAG_DONE );
  652.                 }
  653.             return;
  654.          FI
  655.          if (aircraft_here) {
  656.             if (PLAYER.show & SHOW_REQ) {
  657.                 playSound(DONK_SOUND,PLAYER.snd_vol);
  658.                 (void)rtEZRequestTags(\
  659.                    "Ground forces cannot attack aircraft that are over water.",
  660.                    "Drat!",NULL,NULL,
  661.                    RT_Window,        map_window,
  662.                    RT_ReqPos,        REQPOS_CENTERSCR,
  663.                    RT_LockWindow,    TRUE,
  664.                    RTEZ_Flags,       EZREQF_CENTERTEXT,
  665.                    TAG_DONE );
  666.                 }
  667.             return;
  668.          FI
  669.          // if weve made it this far, there must only be ships
  670.          if (wishbook[defender->type].ship_flag)
  671.             bombardment = TRUE;
  672.       FI
  673.  
  674.    // determine whether the attacking unit has enough movement value left
  675.    // to carry out the assault, i.e. enough to enter the enemy hex area
  676.    if (def_terrain==HEX_CITY || bombardment==TRUE)
  677.       movement_cost = 60;
  678.    else
  679.       movement_cost = movement_cost_table[attacker->type][def_terrain];
  680.  
  681.    if (movement_cost < attacker->move) {
  682.       attacker->move -= movement_cost;
  683.       if (attacker->move<11) {
  684.          attacker->move = 0;
  685.          control_flag = UNIT_DONE;
  686.       }
  687.    } else {    // he only has enough movement for a /chance/ at this
  688.       int chance = attacker->move;
  689.  
  690.       attacker->move = 0;
  691.       control_flag = UNIT_DONE;
  692.       if (RangeRand(movement_cost)>chance) {
  693.          if( PLAYER.soundfx == SOUND_ALL ) {
  694.              // Only play the sound if the player wants stuff shown
  695.              playSound(SMASH_SOUND,PLAYER.snd_vol);
  696.          }
  697.          return;     // his movement attempt failed
  698.       FI
  699.    }
  700.  
  701.    /*
  702.       Bombardment, in the context of this game, means that a unit is attacking
  703.       a hex area that it cannot move onto, such as a cruiser firing upon rifle
  704.       units.  The only difference between bombardment and normal combat is that
  705.       the victorious attacking unit does not move.
  706.    */
  707.  
  708.    /*
  709.       Now we take time out to initialize and record basic information
  710.       into the structure for the combat report.
  711.    */
  712.    battle.turn = turn;
  713.    battle.att_x = attacker->col;
  714.    battle.att_y = attacker->row;
  715.    battle.att_owner = attacker->owner;
  716.    battle.att_type = attacker->type;
  717.    battle.def_x = defender->col;
  718.    battle.def_y = defender->row;
  719.    battle.def_owner = defender->owner;
  720.    battle.white_icon = white_flag;
  721.    battle.def_type = defender->type;
  722.    battle.winner = 0;
  723.    battle.casualties = 0;
  724.    battle.bombardment = bombardment;
  725.    // always seen by defender; we can add others flags later
  726.    battle.seen_by = mask(battle.def_owner);
  727.    battle.blows = 0L;
  728.  
  729.    // track number of attacks made by this unit
  730.    attacker->attacks++;
  731.  
  732.    // move the unit
  733.    if (bombardment==FALSE) {
  734.       attacker->col = targx;
  735.       attacker->row = targy;
  736.    FI
  737.    // keep survey mode up to date
  738.    cursx = attacker->col;  cursy = attacker->row;
  739.    if (attacker->ship) {     // he must be attacking from on board a ship
  740.       attacker->ship->cargo--;
  741.       attacker->ship->weight -= cargo_weight(attacker->type);
  742.       if (attacker->ship->type==TRANSPORT)
  743.          attacker->move = 0;   // unloading from a transport expends movement
  744.       attacker->ship=NULL;  // so we unlink him from it
  745.    FI
  746.    if (attacker->type==TRANSPORT && attacker->cargo>0) {   // ship moves, cargo moves
  747.       struct Unit *cargo=(struct Unit *)unit_list.mlh_Head;
  748.       for (; cargo->unode.mln_Succ; cargo=(struct Unit *)cargo->unode.mln_Succ)
  749.          if (cargo->ship==attacker) {
  750.             cargo->col=targx;  cargo->row=targy;
  751.             clear_orders(cargo);
  752.          }
  753.    }
  754.    if (wishbook[attacker->type].range>0)   // units that need fuel, i.e. aircraft
  755.       attacker->fuel--;
  756.  
  757.    /*
  758.       This function looks at an attacking unit and a defending unit and adds up
  759.       various modifiers based on the types of units, the terrain, the handicap
  760.       of the players, etc.  It sticks the results in attfact and deffact.
  761.    */
  762.    NCS_combat_mods(attacker,defender,&attfact,&deffact);
  763.  
  764.    /*
  765.       battle actually takes place; calculate attacks, damage
  766.       record combat reports, display battle action and sounds
  767.       NOTE: battle always begins with two visible blows that have no game
  768.       effect, but serve to establish visually who is attacking who
  769.    */
  770.    num_blows = 2L;    bit_blows = 2L;
  771.    while (ALIVE(attacker) && ALIVE(defender)) {
  772.       int attval, defval;
  773.       static int strength[] = { 1, 1, 1, 2, 1, 1, 3, 1, 2, 3, 1, 1 };
  774.  
  775.       /*
  776.          strength is the amount of damage done by a hit from the unit
  777.          infantry --- 1
  778.          armor ------ 1
  779.          aircav ----- 1
  780.          bomber ----- 2
  781.          fighter ---- 1
  782.          transport -- 1
  783.          sub -------- 3
  784.          destroyer -- 1
  785.          cruiser ---- 2
  786.          battleship - 3
  787.          carrier ---- 1
  788.       */
  789.  
  790.       /*
  791.          Here is the NCS (New Combat System) which is based on simulated roll of
  792.          three six-sided dice (3d6) for each player.
  793.  
  794.             attval = 3d6 + attfact
  795.             defval = 3d6 + deffact
  796.  
  797.          And whoever's total is higher wins the round of combat.
  798.       */
  799.  
  800.       do {
  801.          attval = 3 + RangeRand(6L) + RangeRand(6L) + RangeRand(6L) + attfact;
  802.          defval = 3 + RangeRand(6L) + RangeRand(6L) + RangeRand(6L) + deffact;
  803.       } while (attval==defval);     // we don't allow draws
  804.  
  805.       if (num_blows<24L) {
  806.          num_blows++;
  807.          bit_blows <<= 1;  // make way for the bit
  808.          if (attval>defval)
  809.             bit_blows |= 1L;   // this bit shows who was hit
  810.       FI
  811.       if (attval>defval)
  812.          defender->damage += strength[attacker->type];
  813.       else {
  814.          attacker->damage += strength[defender->type];
  815.  
  816.          /*
  817.             If the attacker is a ship, and it's taken enough damage to slow its
  818.             speed, then I want to go back and retro-actively reduce its rate for
  819.             THIS turn.  In other words, if my destroyer (speed 3) attacks on its
  820.             first move of this turn, and it takes two points of damage, it's
  821.             speed should be *immediately* reduced to one -- and that one has
  822.             already been expended in the attack!
  823.          */
  824.          if (unit_speed(attacker)<wishbook[attacker->type].speed)
  825.             attacker->move = 0;
  826.       FI
  827.    OD
  828.    battle.blows = (num_blows<<24) | bit_blows;
  829.  
  830.    if (ALIVE(defender)) {   // defeat first; its easier
  831.       if (attacker->ship) {  // he could be attacking from on board a ship!
  832.          attacker->ship->cargo--;
  833.          attacker->ship->weight -= cargo_weight(attacker->type);
  834.       FI
  835.       Remove((struct Node *)attacker);
  836.       // increment Units Lost in Combat
  837.       roster[attacker->owner].ulc[attacker->type]++;
  838.       roster[defender->owner].eud[attacker->type]++;
  839.       control_flag = UNIT_LOST;
  840.       /* record the event for posterity */
  841.       battle.winner = battle.def_owner;
  842.    } else {    // victory here
  843.       // go down the list for defending units to destroy
  844.       // we must do multiple searches, because destroying a unit messes
  845.       // up the list structure
  846.       BOOL unit_found;
  847.       struct Unit *unit;
  848.       do {
  849.          unit_found = FALSE;
  850.          for (unit=(struct Unit *)unit_list.mlh_Head;unit->unode.mln_Succ;unit=(struct Unit *)unit->unode.mln_Succ)
  851.             if (unit->col==targx && unit->row==targy && unit->owner!=attacker->owner) {
  852.                if (attacker->type==FIGHTER && bombardment==TRUE)
  853.                   if (wishbook[unit->type].range<=0)  // not an aircraft
  854.                      continue;
  855.                Remove((struct Node *)unit);
  856.                roster[unit->owner].ulc[unit->type]++;    // inc. Units Lost in Combat
  857.                roster[attacker->owner].eud[unit->type]++;   // inc. Enemy Units Destroyed
  858.                battle.casualties++;
  859.                destruct_unit(unit);
  860.                unit_found = TRUE;
  861.                break;
  862.             FI
  863.       } while (unit_found);
  864.       // if theres a city here, attacker conquers it
  865.       captured = city_hereP(targx,targy);
  866.       if (captured)
  867.          if (captured->owner)    // a non-neutral city is taken
  868.             battle.seen_by |= mask(captured->owner);
  869.       if (captured!=NULL && bombardment==FALSE) {
  870.          // attacking unit is dispersed to hold the city
  871.          Remove((struct Node *)attacker);
  872.          roster[attacker->owner].ulc[attacker->type]++;   // inc. Units Lost in Combat
  873.          control_flag = UNIT_LOST;
  874.       FI
  875.       battle.winner = battle.att_owner;
  876.    FI
  877.  
  878.    // display the battle
  879.    if (ISHUMAN(PLAYER.type))  show_battle();
  880.    if (captured!=NULL && bombardment==FALSE)
  881.       conquer_city(captured);
  882.       // BSH 1/10 - Removed the check for human player -
  883.       //        always call conquer_city - it checks inside
  884.       //        for human player and calls AI when needed
  885.  
  886.    // clean up some things left over after movement & battle
  887.    if (control_flag!=UNIT_LOST)
  888.       if (wishbook[attacker->type].range>0 && attacker->fuel<=0) {
  889.          // aircraft out of fuel, crashes
  890.          Remove((struct Node *)attacker);
  891.          roster[attacker->owner].ulc[attacker->type]++;
  892.          control_flag = UNIT_LOST;
  893.       FI
  894.    if (control_flag==UNIT_LOST)
  895.       destruct_unit(attacker);   // destruct and free RAM
  896.    {
  897.       BOOL vis=Bool(roster[att_player].show&SHOW_GRP);
  898.       explore_at_hex(att_player,targx,targy,vis,TRUE);
  899.    }
  900.    if (def_player>0)    // neutral cities never explore
  901.       explore_at_hex(def_player,targx,targy,INVISIBLE,TRUE);
  902.  
  903.    if (control_flag!=UNIT_LOST)
  904.       if (roster[attacker->owner].show & SHOW_GRP)  // Was SHOW_REQ ****
  905.          unit_status_bar(attacker);
  906.  
  907.    {  // write the combat report file
  908.       BPTR outfile;
  909.  
  910.       strcpy(foo,prefix);
  911.       strcat(foo,id_filetag);
  912.       strcat(foo,".CR");
  913.       if (outfile = Open(foo,MODE_OLDFILE)) {
  914.          Seek(outfile,0L,OFFSET_END);
  915.          Write(outfile,&battle,sizeof(battle));
  916.          Close(outfile);
  917.       FI
  918.    }
  919. }
  920.  
  921.  
  922. /*
  923.    This is part of the NCS (New Combat System).
  924.  
  925.    This NCS function will accept the information about a combat event and
  926.    calculate the attack and defense modifiers for both units -- taking into
  927.    account unit types, terrain, stacking, and the global handicap values of
  928.    the two players.
  929. */
  930.  
  931. void NCS_combat_mods(attacker, defender, att_mod, def_mod)
  932. struct Unit *attacker, *defender;
  933. int *att_mod, *def_mod;
  934. {
  935.    int terrain = get(t_grid,defender->col,defender->row);
  936.    int atype = attacker->type;
  937.    int dtype = defender->type;
  938.  
  939.    // I can do this shortcut because airfields defend like infantry in
  940.    // every respect; of course it is only changed locally in this function!
  941.    if (dtype==AIRBASE)
  942.       dtype = RIFLE;
  943.  
  944.    // start with the global handicap values of the players
  945.    *att_mod = roster[attacker->owner].att;
  946.    *def_mod = roster[defender->owner].def;
  947.  
  948.    // transports always have a -4 on defense
  949.    if (dtype==TRANSPORT)
  950.       *def_mod -= 4;
  951.  
  952.    // transports and carriers have a -4 on attack
  953.    if (atype==TRANSPORT || atype==CARRIER)
  954.       *att_mod -= 4;
  955.  
  956.    // carriers have a -2 defense and -1 for every 2 fighters on board
  957.    if (dtype==CARRIER) {
  958.       *def_mod -= 2;
  959.       {      // count units on board
  960.          int ctr = 0;
  961.          struct Unit *unit = (struct Unit *)unit_list.mlh_Head;
  962.          for (; unit->unode.mln_Succ; unit = (struct Unit *)unit->unode.mln_Succ)
  963.             if (unit->ship==defender) ctr++;
  964.          *def_mod -= ctr/2;
  965.       }
  966.    }
  967.  
  968.    // submarines have a basic -3 defense
  969.    if (dtype==SUB)
  970.       *def_mod -= 3;
  971.  
  972.    // fighters & bombers have a +3 defending against ships
  973.    if (dtype==FIGHTER || dtype==BOMBER)
  974.       if (wishbook[atype].ship_flag)
  975.          *def_mod += 3;
  976.  
  977.    // fighters have a +3 attack against bombers or aircav
  978.    if (dtype==BOMBER || dtype==AIRCAV)
  979.       if (atype==FIGHTER)
  980.          *att_mod += 3;
  981.  
  982.    /*
  983.       ground units defending:
  984.          -3 against naval bombardment
  985.          -2 against bombers
  986.          -1 against other aircraft
  987.    */
  988.    if (dtype==RIFLE || dtype==ARMOR) {
  989.       switch (dtype) {
  990.          case CRUISER:
  991.          case BATTLESHIP:
  992.             *def_mod--;
  993.          case BOMBER:
  994.             *def_mod--;
  995.          case FIGHTER:
  996.          case AIRCAV:
  997.             *def_mod--;
  998.       }
  999.    }
  1000.  
  1001.    /* That takes care of the unit-based modifiers.  Now the terrain stuff! */
  1002.    switch (terrain) {
  1003.  
  1004.       case HEX_DESERT:
  1005.          // ground forces -2 defense against aircraft
  1006.          if (dtype==RIFLE || dtype==ARMOR)
  1007.             if (wishbook[atype].range>0)
  1008.                *def_mod -= 2;
  1009.          // infantry -2 defense against armor
  1010.          if (dtype==RIFLE && atype==ARMOR)
  1011.             *def_mod -= 2;
  1012.          break;
  1013.  
  1014.       case HEX_BRUSH:
  1015.          // ground forces +1 defense against fighters or bombers
  1016.          if (dtype==RIFLE || dtype==ARMOR)
  1017.             if (atype==FIGHTER || atype==BOMBER)
  1018.                *def_mod++;
  1019.          break;
  1020.  
  1021.       case HEX_FOREST:
  1022.          // ground forces +2 defense against fighters or bombers
  1023.          if (dtype==RIFLE || dtype==ARMOR)
  1024.             if (atype==FIGHTER || atype==BOMBER)
  1025.                *def_mod += 2;
  1026.          // infantry +2 defense against armor
  1027.          if (dtype==RIFLE && atype==ARMOR)
  1028.             *def_mod += 2;
  1029.          // armor -2 defense against infantry
  1030.          if (dtype==ARMOR && atype==RIFLE)
  1031.             *def_mod -= 2;
  1032.          break;
  1033.  
  1034.       case HEX_JUNGLE:
  1035.          // ground forces +3 defense against fighters or bombers
  1036.          if (dtype==RIFLE || dtype==ARMOR)
  1037.             if (atype==FIGHTER || atype==BOMBER)
  1038.                *def_mod += 3;
  1039.          // infantry +3 defense against armor
  1040.          if (dtype==RIFLE && atype==ARMOR)
  1041.             *def_mod += 3;
  1042.          // armor -3 defense against infantry
  1043.          if (dtype==ARMOR && atype==RIFLE)
  1044.             *def_mod -= 3;
  1045.          break;
  1046.  
  1047.       case HEX_SWAMP:
  1048.          // ground forces +1 defense against aircraft
  1049.          if (dtype==RIFLE || dtype==ARMOR)
  1050.             if (wishbook[atype].range>0)
  1051.                *def_mod++;
  1052.          // infantry +2 defense against armor shelling
  1053.          if (dtype==RIFLE && atype==ARMOR)
  1054.             *def_mod += 2;
  1055.          break;
  1056.  
  1057.       case HEX_RUGGED:
  1058.          // ground forces, +1 defending against fighter or bomber
  1059.          if (dtype==RIFLE || dtype==ARMOR)
  1060.             if (atype==FIGHTER || atype==BOMBER)
  1061.                *def_mod++;
  1062.          break;
  1063.  
  1064.       case HEX_HILLS:
  1065.          // ground forces, +2 defending against fighter or bomber
  1066.          // ground forces, +1 defending against cruiser or battleship
  1067.          if (dtype==RIFLE || dtype==ARMOR) {
  1068.             if (atype==FIGHTER || atype==BOMBER)
  1069.                *def_mod += 2;
  1070.             if (atype==CRUISER || atype==BATTLESHIP)
  1071.                *def_mod += 1;
  1072.          }
  1073.          break;
  1074.  
  1075.       case HEX_MOUNTAINS:
  1076.          // ground forces, +3 defending against fighter or bomber
  1077.          // ground forces, +2 defending against air cavalry
  1078.          // ground forces, +2 defending against cruiser or battleship
  1079.          if (dtype==RIFLE || dtype==ARMOR) {
  1080.             if (atype==FIGHTER || atype==BOMBER)
  1081.                *def_mod += 3;
  1082.             if (atype==AIRCAV)
  1083.                *def_mod += 2;
  1084.             if (atype==CRUISER || atype==BATTLESHIP)
  1085.                *def_mod += 2;
  1086.          }
  1087.          // infantry defending +3 against armor shelling
  1088.          if (dtype==RIFLE && atype==ARMOR)
  1089.             *def_mod += 3;
  1090.          break;
  1091.  
  1092.       case HEX_SHALLOWS:
  1093.          // subs attack -3 against ships
  1094.          if (atype==SUB && wishbook[dtype].ship_flag)
  1095.             *att_mod -= 3;
  1096.          // subs defend -2 (on top of the -3 they already have!)
  1097.          if (dtype==SUB)
  1098.             *def_mod -= 2;
  1099.          // all other ships defend at -1 in shallows
  1100.          if (wishbook[dtype].ship_flag && dtype!=SUB)
  1101.             *def_mod--;
  1102.          break;
  1103.  
  1104.       case HEX_DEPTH:
  1105.          // sub defends at +3 in depths
  1106.          if (dtype==SUB)
  1107.             *def_mod += 3;
  1108.          break;
  1109.    }
  1110. }
  1111.  
  1112.  
  1113. /* This routine determines the basic attack values.  I'm just passing in
  1114.    values instead of unit pointers so I can use this later for the AI
  1115.    player(s) to do "what if" scenarios and determine what is risky/not
  1116.    risky attack.
  1117.    */
  1118. void   get_attack_values ( short att_type, int att_terrain, int att_player,
  1119.        short def_type, int def_terrain, int def_player,
  1120.        int* attfact, int* deffact)
  1121. {
  1122.     /*was attfact = 100; */
  1123.    // Now we use the attack factor from the start game setting for the player
  1124.    *attfact = roster[att_player].att;
  1125.    if (att_type==TRANSPORT || att_type==CARRIER)
  1126.       *attfact /= 2;
  1127.  
  1128.    /*was deffact = 100; */
  1129.    // Now we use the defense factor from the start game settings
  1130.    *deffact = roster[def_player].def;
  1131.    if (def_type==TRANSPORT || def_type==SUB)
  1132.       *deffact *= 6/10;
  1133.    if (def_type==RIFLE && (att_type==CRUISER || att_type==BATTLESHIP))
  1134.       *deffact *= 6/10;
  1135.    if (def_type==BOMBER && att_type==FIGHTER)
  1136.       *deffact *= 6/10;
  1137.  
  1138.    // terrain effects modify attack and defense factors
  1139.    switch (att_type) {
  1140.       case ARMOR:
  1141.       case BOMBER:
  1142.          if (def_terrain==HEX_DESERT)
  1143.             *attfact += 25;
  1144.          break;
  1145.       case AIRCAV:
  1146.          if (def_terrain==HEX_FOREST)
  1147.             *attfact += 10;
  1148.          if (def_terrain==HEX_JUNGLE)
  1149.             *attfact += 20;
  1150.          break;
  1151.    }
  1152.    if (def_type==RIFLE) {
  1153.       if (def_terrain==HEX_FOREST)
  1154.          *deffact += 10;
  1155.       if (def_terrain==HEX_JUNGLE)
  1156.          *deffact += 20;
  1157.    }
  1158.  
  1159.    // altitude affects odds between ground units (never ships or planes)
  1160.    if (att_type==RIFLE || att_type==ARMOR)
  1161.       if (def_type==RIFLE || def_type==ARMOR) {
  1162.          switch (att_terrain) {
  1163.             case HEX_MOUNTAINS:   // these are cumulative
  1164.                *attfact += 10;    // neat, huh?
  1165.             case HEX_HILLS:
  1166.                *attfact += 10;
  1167.             case HEX_RUGGED:
  1168.                *attfact += 10;
  1169.          }
  1170.          switch (def_terrain) {
  1171.             case HEX_MOUNTAINS:
  1172.                *deffact += 10;
  1173.             case HEX_HILLS:
  1174.                *deffact += 10;
  1175.             case HEX_RUGGED:
  1176.                *deffact += 10;
  1177.          }
  1178.       FI
  1179.    /* End if */
  1180. }
  1181.  
  1182.  
  1183. // this is where each player starts the game; it selects the home city, sets
  1184. // initial production, etc.
  1185. //  Broken up into two routines so that the player will not be surprised
  1186. void create_initial_city()
  1187. {  // put a player on the map!
  1188.  
  1189.    struct City *home_city;    /* this player's starting city */
  1190.  
  1191.    // STEP ONE: locate a home city for him
  1192.    do {
  1193.       struct City *metro = (struct City *)city_list.mlh_Head;
  1194.       long num_cities = count_nodes(&city_list);
  1195.       int home_city_index = RangeRand(num_cities)+1;
  1196.       int ctr;
  1197.  
  1198.       for ( ctr = 1; metro->cnode.mln_Succ; metro = (struct City *)metro->cnode.mln_Succ, ctr++)
  1199.          if (ctr == home_city_index) {
  1200.             home_city = metro;
  1201.             break;
  1202.          FI
  1203.    } while (home_city->owner);  /* make sure it's a neutral city */
  1204.  
  1205.    // Make him the owner
  1206.    home_city->owner = player;
  1207.  
  1208.    return;
  1209. }
  1210. void jumpstart_player()
  1211. {
  1212.     /* Find the player's home city */
  1213.    struct City* metro = (struct City*)city_list.mlh_Head;
  1214.    while( (metro->cnode.mln_Succ) && (metro->owner != player) )
  1215.       metro = (struct City*)metro->cnode.mln_Succ;
  1216.  
  1217.    // create his map display, all spaces unknown
  1218.    if (alloc_map(&PLAYER.map)==FALSE)
  1219.       clean_exit(2,"ERROR: Fatal RAM allocation disaster!\n");
  1220.  
  1221.    // put his city and surrounding area on the map
  1222.    explore_at_hex(player,metro->col,metro->row,INVISIBLE,TRUE);
  1223.  
  1224.    // If File-Mail play create passwords
  1225.    //if(fmail)
  1226.    //     create_passlock();
  1227.  
  1228.    // call him to the console and show the map
  1229.    create_player_display(metro->col,metro->row);
  1230.  
  1231.    // initialize his city production
  1232.    conquer_city(metro);
  1233. }
  1234.  
  1235.  
  1236. void unit_name_request(metro,unit)
  1237. struct City *metro;
  1238. struct Unit *unit;
  1239. {
  1240.    BOOL exam=FALSE, ship=wishbook[unit->type].ship_flag;
  1241.    char *name;
  1242.    struct Window *name_window=NULL;
  1243.    struct Gadget *context, *name_gad, *cont_gad, *prod_gad;
  1244.    struct NewGadget button = {
  1245.       77,66,      // leftedge, topedge
  1246.       98,16,   // width, height
  1247.       "_Continue",  // text label
  1248.       NULL,    // font
  1249.       1,       // gadget ID
  1250.       NULL,NULL,NULL
  1251.    }, strfield = {
  1252.       173,40,      // leftedge, topedge
  1253.       200,16,   // width, height
  1254.       "Ship _Name:", // text label
  1255.       NULL,    // font
  1256.       3,       // gadget ID
  1257.       PLACETEXT_LEFT,
  1258.       NULL,NULL
  1259.    };
  1260.  
  1261.    /* make sure user doesn't play with the map window now */
  1262.    SetPointer(map_window,BUSY_POINTER);
  1263.    ModifyIDCMP(map_window,NULL);
  1264.  
  1265.    // create the [Continue] and [Production] buttons
  1266.    if (!CreateContext(&context))
  1267.       clean_exit(1,"Unable to create context gadget!");
  1268.    button.ng_VisualInfo = vi;
  1269.    button.ng_TextAttr = &topaz11bold;
  1270.    cont_gad = CreateGadget(BUTTON_KIND,context,&button,
  1271.       GT_Underscore, '_',
  1272.       TAG_END);
  1273.    button.ng_LeftEdge = 221;
  1274.    button.ng_GadgetText = "_Production";
  1275.    button.ng_TextAttr = &topaz11;
  1276.    button.ng_Flags = NULL;
  1277.    prod_gad = CreateGadget(BUTTON_KIND,cont_gad,&button,
  1278.       GT_Underscore, '_',
  1279.       TAG_END);
  1280.  
  1281.    // randomly select a name for this unit
  1282.    name = random_name(unit->type);
  1283.  
  1284.    // create the name field
  1285.    strfield.ng_VisualInfo = vi;
  1286.    strfield.ng_TextAttr = &topaz11;
  1287.    if (ship)
  1288.    name_gad = CreateGadget(STRING_KIND,prod_gad,&strfield,
  1289.       GTST_String,            name,
  1290.       GTST_MaxChars,          19L,
  1291.       STRINGA_Justification,  GACT_STRINGCENTER,
  1292.       GT_Underscore,          '_',
  1293.       TAG_END);
  1294.  
  1295.    // do the window itself
  1296.    {
  1297.       int x, y, wd=414, ht=88;
  1298.  
  1299.       // I need to do some pixelly math to figure out the most
  1300.       // aesthetically pleasing place to put the requester
  1301.       // It must be near the city in question, yet not hide it from view.
  1302.       log_to_abs(metro->col,metro->row,&x,&y);
  1303.       x -= 30;
  1304.       y += 45;
  1305.       if (y>map_window->Height-ht)
  1306.          y -= (90+ht);
  1307.  
  1308.       name_window = OpenWindowTags(NULL,
  1309.          WA_Gadgets,       context,
  1310.          WA_Title,         "New unit produced...",
  1311.          WA_CustomScreen,  map_screen,
  1312.          WA_Top,y,         WA_Left,x,
  1313.          WA_Height,ht,     WA_Width,wd,
  1314.          WA_IDCMP,         IDCMP_GADGETUP|IDCMP_VANILLAKEY,
  1315.          WA_Flags,         NOCAREREFRESH|WFLG_DRAGBAR|WFLG_ACTIVATE,
  1316.          TAG_END );
  1317.    }
  1318.    if (name_window==NULL)
  1319.       clean_exit(1,"ERROR: Unable to open production requester!");
  1320.  
  1321.    // render in some text and a little picture
  1322.    rast_port = name_window->RPort;
  1323.    sprintf(foo,"%s produced a %s.",metro->name,wishbook[unit->type].name);
  1324.    plot_text(10,21,foo,BLACK,LT_GRAY,JAM2,&topaz11);
  1325.    px_plot_icon(unit->type,45,39,PLAYER.color,0,FALSE);
  1326.    if (!ship) {
  1327.       sprintf(foo,"%s %s",name,wishbook[unit->type].name);
  1328.       plot_text(125,43,foo,BLACK,LT_GRAY,JAM2,&topaz11);
  1329.    FI
  1330.  
  1331.    {  // handle the user actions here
  1332.       struct IntuiMessage *message; // the message the IDCMP sends us
  1333.  
  1334.       // useful for interpreting IDCMP messages
  1335.       UWORD code;
  1336.       ULONG class;
  1337.       APTR object;
  1338.       UWORD qualifier;
  1339.  
  1340.       FOREVER {
  1341.          WaitPort(name_window->UserPort);
  1342.          while (message = GT_GetIMsg(name_window->UserPort)) {
  1343.             code = message->Code;  // MENUNUM
  1344.             object = message->IAddress;  // Gadget
  1345.             class = message->Class;
  1346.             qualifier = message->Qualifier;
  1347.             GT_ReplyIMsg(message);
  1348.             if (class==IDCMP_VANILLAKEY) {
  1349.                switch ((char)code) {
  1350.                   case 13:
  1351.                   case 'c':    // default for CONTINUE
  1352.                      // show the button depressed
  1353.                      show_depress(cont_gad,name_window->RPort);
  1354.                      Delay(10L);
  1355.                      goto exit_name_window;
  1356.                   case 'p':   // default for PRODUCTION
  1357.                      show_depress(prod_gad,name_window->RPort);
  1358.                      Delay(10L);
  1359.                      exam = TRUE;
  1360.                      goto exit_name_window;
  1361.                   case 'n':   // activate the NAME string
  1362.                      ActivateGadget(name_gad,name_window,NULL);
  1363.                }
  1364.             FI
  1365.             if (class==IDCMP_GADGETUP) {
  1366.                if (object==cont_gad) {
  1367.                   goto exit_name_window;
  1368.                FI
  1369.                if (object==prod_gad) {
  1370.                   exam = TRUE;
  1371.                   goto exit_name_window;
  1372.                FI
  1373.             FI
  1374.          OD
  1375.       OD
  1376.    }
  1377.    exit_name_window:
  1378.  
  1379.    // fetch new name string from the gadget
  1380.    if (ship)
  1381.       name_unit(unit,((struct StringInfo *)name_gad->SpecialInfo)->Buffer);
  1382.    else
  1383.       name_unit(unit,name);
  1384.  
  1385.    // now close up everything with the mapsize_window
  1386.    CloseWindow(name_window);
  1387.    rast_port = map_window->RPort;
  1388.    name_window = NULL;
  1389.    FreeGadgets(context);
  1390.    ClearPointer(map_window);
  1391.    ModifyIDCMP(map_window,IDCMP_PLAYGAME);
  1392.    if (exam)
  1393.       examine_city(metro);
  1394. }
  1395.  
  1396.  
  1397. // go through the production of all cities owned by the current player, and
  1398. // do this turn's production of military units, etc.
  1399. // it returns the number of units produced
  1400.  
  1401. int do_cities_production()
  1402. {
  1403.    struct City *metro = (struct City *)city_list.mlh_Head;
  1404.    int ctr = 0;
  1405.  
  1406.    /*
  1407.         The map wasn't updating correctly sometimes, so I decided to make
  1408.         two passes on the city list.  Now it goes through the list once to
  1409.         update the WIP and the map area around the cities, then a second
  1410.         pass for actual production of units.
  1411.    */
  1412.  
  1413.    for( ; metro->cnode.mln_Succ; metro=(struct City*)metro->cnode.mln_Succ)
  1414.       if( metro->owner == player) {
  1415.          metro->unit_wip += (metro->industry*PLAYER.prod)/50;
  1416.          explore_at_hex(player,metro->col,metro->row,INVISIBLE,TRUE);
  1417.       }
  1418.  
  1419.    metro = (struct City*)city_list.mlh_Head;
  1420.    for ( ; metro->cnode.mln_Succ; metro = (struct City *)metro->cnode.mln_Succ)
  1421.       if (metro->owner==player) {
  1422.         if (metro->unit_wip >= wishbook[metro->unit_type].build) {
  1423.             struct Unit *new_unit = AllocVec((int)sizeof(*new_unit),MEMF_CLEAR);
  1424.  
  1425.             metro->unit_wip = 0;
  1426.  
  1427.             // build the unit
  1428.             new_unit->col = metro->col;
  1429.             new_unit->row = metro->row;
  1430.             new_unit->owner = player;
  1431.             new_unit->type = metro->unit_type;
  1432.             new_unit->move = unit_speed(new_unit);
  1433.             new_unit->damage = 0;
  1434.  
  1435.             // Added by BSH
  1436.             new_unit->attacks = 0;
  1437.             new_unit->cargo = 0;
  1438.             new_unit->ship = NULL;
  1439.             new_unit->orders = NULL;
  1440.             // End Added BSH
  1441.  
  1442.             new_unit->fuel = wishbook[new_unit->type].range;
  1443.             name_unit(new_unit,"UNNAMED");
  1444.  
  1445.             AddTail((struct List *)&unit_list,(struct Node *)new_unit);
  1446.             ctr++;
  1447.  
  1448.             // now inform the user and let him reset the production
  1449.             if (wishbook[new_unit->type].ship_flag || PLAYER.show_production) {
  1450.                if (display==FALSE)
  1451.                   create_player_display(metro->col,metro->row);
  1452.                if (need_to_scrollP(metro->col,metro->row)) {
  1453.                   int ox=xoffs, oy=yoffs;
  1454.  
  1455.                   set_display_offsets(metro->col,metro->row);
  1456.                   GP_smart_scroll(ox,oy);
  1457.                FI
  1458.  
  1459.                // do this so the user can see which city he is looking at
  1460.                save_hex_graphics(metro->col,metro->row,0);    // blit the background to a safe place
  1461.                plot_mapobject(metro->col,metro->row,MAP_MARKER);
  1462.                unit_name_request(metro,new_unit);
  1463.  
  1464.                // unmark the city on screen
  1465.                GP_update_hex_display(metro->col,metro->row);
  1466.             FI
  1467.          FI
  1468.       FI
  1469.    return ctr;
  1470. }
  1471.  
  1472.  
  1473. void unit_status_bar(unit)
  1474. struct Unit *unit;
  1475. {
  1476.     char foo2[30];
  1477.    /*
  1478.       With ships, the unit type goes first, like for example...
  1479.          Battleship Texas
  1480.       ...but with other units the name goes first, then the type...
  1481.          101st Infantry
  1482.    */
  1483.    if (wishbook[unit->type].ship_flag)
  1484.       sprintf(foo,"%s %s",wishbook[unit->type].name,unit->name);
  1485.    else
  1486.       sprintf(foo,"%s %s",unit->name,wishbook[unit->type].name);
  1487.    if (wishbook[unit->type].hitpoints>1) {
  1488.       sprintf(bar,"  hits:%ld/%ld",unit->damage,wishbook[unit->type].hitpoints);
  1489.       strcat(foo,bar);
  1490.    FI
  1491.    if (wishbook[unit->type].range>0) {
  1492.       sprintf(bar,"  fuel:%ld/%ld",unit->fuel,wishbook[unit->type].range);
  1493.       strcat(foo,bar);
  1494.    FI
  1495.    if (unit->type==TRANSPORT || unit->type==CARRIER) {
  1496.       sprintf(bar,"  cargo:%ld/%ld",unit->cargo,(unit->type==TRANSPORT)?6:8);
  1497.       strcat(foo,bar);
  1498.    FI
  1499.    if (unit->type==AIRBASE) {
  1500.       sprintf(bar," on-base: %ld aircraft",unit->cargo);
  1501.       strcat(foo,bar);
  1502.    FI
  1503.    if (unit->orders)
  1504.       switch (unit->orders->type) {
  1505.          case ORDER_LOAD:
  1506.             strcat(foo,"   orders: Load Ship");
  1507.             break;
  1508.          case ORDER_SENTRY:
  1509.             strcat(foo,"   orders: Sentry Duty");
  1510.             break;
  1511.          case ORDER_FORTIFY:
  1512.             strcat(foo,"   orders: Dig In");
  1513.             break;
  1514.          case ORDER_UNLOAD:
  1515.            strcat(foo,"   status: Unloading Units");
  1516.            break;
  1517.          case ORDER_FORTIFIED:
  1518.            strcat(foo,"   status: In Fortified Position");
  1519.            break;
  1520.          case ORDER_GOTO:
  1521.            if( !unit->orders->dest_unit ) {
  1522.              if( unit->orders->etc >= 0 ) {
  1523.                 strcat(foo,"   orders: Goto Hex: ");
  1524.                 sprintf(foo2, "%ld, %ld", unit->orders->destx,
  1525.                      unit->orders->desty);
  1526.                 strcat(foo, foo2);
  1527.              }
  1528.              else {
  1529.                 strcat(foo,"   orders: Patrol Between: ");
  1530.                 sprintf(foo2, "%ld, %ld and %ld, %ld",
  1531.                     unit->orders->destx, unit->orders->desty,
  1532.                     unit->orders->orgx, unit->orders->orgy);
  1533.                 strcat(foo, foo2);
  1534.              }
  1535.            }
  1536.            else {
  1537.              strcat(foo,"   orders: Rendevous with Ship: ");
  1538.              if( AI3_AssertUnit( unit->orders->dest_unit ) )
  1539.                 strcat(foo, unit->orders->dest_unit->name);
  1540.            }
  1541.            break;
  1542.       } // end switch
  1543.  
  1544.    strncpy(win_title,foo,79L);
  1545.    SetWindowTitles(map_window,win_title,(UBYTE *)~0);
  1546.  
  1547.    // now do the unit movement bar
  1548.    {
  1549.       int barleft, barright;
  1550.       int bartop, barbottom, barheight;
  1551.       int amplitude, ctr;
  1552.  
  1553.       InstallClipRegion(map_window->WLayer,bar_region);
  1554.       SetRast(rast_port,BLACK);
  1555.  
  1556.       // if the unit is an airbase, we're finished with it
  1557.       if (unit->type!=AIRBASE) {
  1558.  
  1559.          // preliminary calculations before I do the graphics
  1560.          bartop = map_window->BorderTop+4;
  1561.          barbottom = map_window->Height-map_window->BorderBottom-15;
  1562.          barleft = map_window->BorderLeft+4;
  1563.          barright = barleft+5;
  1564.          barheight = barbottom-bartop+1;
  1565.          amplitude = (unit->move*barheight)/wishbook[unit->type].speed;
  1566.  
  1567.          // draw the green color bar
  1568.          SetAPen(rast_port,GREEN);
  1569.          if (wishbook[unit->type].range>0) // i.e. an aircraft
  1570.             SetAPen(rast_port,LT_BLUE);
  1571.          if (wishbook[unit->type].ship_flag)
  1572.             SetAPen(rast_port,DK_BLUE);
  1573.  
  1574.          RectFill(rast_port,barleft,barbottom-amplitude,barright,barbottom);
  1575.  
  1576.          // draw the graduation markers
  1577.          SetAPen(rast_port,WHITE);
  1578.          for (ctr=0; ctr<=wishbook[unit->type].speed; ctr+=60) {
  1579.                amplitude = (ctr*barheight)/wishbook[unit->type].speed;
  1580.                Move(rast_port,barleft,barbottom-amplitude);
  1581.                Draw(rast_port,barright,barbottom-amplitude);
  1582.          OD
  1583.  
  1584.       FI
  1585.       InstallClipRegion(map_window->WLayer,map_region);
  1586.    }
  1587. }
  1588.  
  1589.  
  1590. void hex_status_bar(col,row)
  1591. int col, row;
  1592. {
  1593.    struct Unit *unit;
  1594.    struct MapIcon *icon;
  1595.    struct City *metro;
  1596.    // static char *terrain[]={
  1597.    //   "Unexplored Area",
  1598.    //   "Forbidden",
  1599.    //   "Plains",
  1600.    //   "Desert",
  1601.    //   "Scrubland",
  1602.    //   "Forest",
  1603.    //   "Jungle",
  1604.    //   "Rough Country",
  1605.    //   "Hills",
  1606.    //   "Mountains",
  1607.    //   "Mountain Peaks",
  1608.    //   "Swamp",
  1609.    //   "Shallow Waters",
  1610.    //   "Ocean",
  1611.    //   "Deep Ocean",
  1612.    //   "Pack Ice"
  1613.    //};
  1614.  
  1615.    clear_movebar();
  1616.    // look for a city
  1617.    if (metro=city_hereP(col,row)) {
  1618.       strncpy(win_title,metro->name,79L);
  1619.       SetWindowTitles(map_window,win_title,(UBYTE *)~0);
  1620.       return;
  1621.    FI
  1622.    // look for FRIENDLY units (units on ships dont count)
  1623.    for (unit = (struct Unit *)unit_list.mlh_Head; unit->unode.mln_Succ; unit = (struct Unit *)unit->unode.mln_Succ)
  1624.       if (unit->col==col && unit->row==row)
  1625.          if (unit->owner==player && unit->ship==NULL) {
  1626.             // unit belongs to me; let unit_status_bar() handle it
  1627.             unit_status_bar(unit);
  1628.             return;
  1629.          FI
  1630.    // look for HOSTILE units
  1631.    for (icon=(struct MapIcon *)PLAYER.icons.mlh_Head;icon->inode.mln_Succ;icon=(struct MapIcon *)icon->inode.mln_Succ)
  1632.       if (icon->col==col && icon->row==row && icon->owner!=player) {
  1633.          sprintf(foo,"HOSTILE: %s's %s",roster[icon->owner].name,wishbook[icon->type].name);
  1634.          strncpy(win_title,foo,79L);
  1635.          SetWindowTitles(map_window,win_title,(UBYTE *)~0);
  1636.          return;
  1637.       FI
  1638.  
  1639.    strncpy(win_title,terrain_name_table[get(PLAYER.map,col,row)],79L);
  1640.    SetWindowTitles(map_window,win_title,(UBYTE *)~0);
  1641. }
  1642.  
  1643.  
  1644. /*
  1645.    unit "cargo" is attempting to load onto a transport or carrier at this
  1646.    location, or land on an airbase; note that there may be multiple ships
  1647.    in this hex; we must try to find one that can accept the unit.
  1648.    The return value is TRUE if a suitable ship was found.  This does not
  1649.    mean the ship was actually boarded, because it may have been full.
  1650.    The calling function should check the cargo->ship value to find out.
  1651. */
  1652.  
  1653. BOOL board_ship(cargo,col,row)
  1654. struct Unit *cargo;
  1655. int col, row;
  1656. {
  1657.    struct Unit *ship=NULL;
  1658.    int shiptype=NULL;
  1659.    BOOL result = FALSE;
  1660.    int weight=cargo_weight(cargo->type);
  1661.  
  1662.    // determine what kind of ship we need to look for: TRANSPORT for ground
  1663.    // units, AIRBASE for aircraft, or CARRIER for fighters
  1664.    switch (cargo->type) {
  1665.       case ARMOR:
  1666.       case RIFLE:
  1667.          shiptype=TRANSPORT;
  1668.          break;
  1669.       case FIGHTER:
  1670.          {
  1671.             int terrain;
  1672.  
  1673.             // Are we on ocean or land?
  1674.             terrain = get(t_grid,cargo->col,cargo->row);
  1675.             shiptype = AIRBASE;
  1676.             if (OCEAN_P(terrain))
  1677.                shiptype=CARRIER;
  1678.             break;
  1679.          }
  1680.       case BOMBER:
  1681.       case AIRCAV:
  1682.          shiptype=AIRBASE;
  1683.          break;
  1684.       default:
  1685.          return result;
  1686.    };
  1687.  
  1688.    /*
  1689.       search for ships (or airbase) of the appropriate type in this hex
  1690.       we don't need to check who they belong to, the hex must be friendly
  1691.       or this function wouldn't have been called (I hope!)
  1692.    */
  1693.    ship=(struct Unit *)unit_list.mlh_Head;
  1694.    for (; ship->unode.mln_Succ; ship=(struct Unit *)ship->unode.mln_Succ)
  1695.       if (ship->type==shiptype && ship->col==col && ship->row==row) {
  1696.          result = TRUE;
  1697.          // see if there is room on the ship to accept this unit
  1698.          if (weight<=cargo_capacity(ship)-ship->weight) {
  1699.             ship->cargo++;     // take up a cargo slot on the ship
  1700.             ship->weight += weight;
  1701.             cargo->ship=ship;  /* set the cargo unit's new location */
  1702.             cargo->col=ship->col;   cargo->row=ship->row;
  1703.             cargo->move=0;     /* cargo unit's movement this turn is expended */
  1704.             if (shiptype==TRANSPORT)
  1705.                give_orders(cargo,ORDER_SENTRY,0,0,-1);   // auto-sentry them
  1706.             if (shiptype==CARRIER || shiptype==AIRBASE)
  1707.                cargo->fuel = wishbook[cargo->type].range;  // refuel aircraft
  1708.             control_flag=UNIT_DONE;    // so that movement_mode() will know
  1709.             // if this ship is under orders to load units and is now full,
  1710.             // then we should clear the orders for it, like so...
  1711.             if (ship->orders)
  1712.                if (ship->orders->type==ORDER_LOAD && ship->weight>=cargo_capacity(ship))
  1713.                   clear_orders(ship);
  1714.             explore_at_hex(player,col,row,VISIBLE,FALSE);   // update map
  1715.             return result;
  1716.          FI
  1717.       FI
  1718.    return result;
  1719. }
  1720.  
  1721.  
  1722. /*
  1723.    This function determines the cargo capacity of a ship.  It returns the
  1724.    maximum number of units the ship can carry in its current condition
  1725.    (taking damage into account), NOT the amount of free space on the ship.
  1726.  
  1727.    Depending on the value you want, use the formulas:
  1728.       free_space=cargo_capacity(ship)-ship->cargo;
  1729.    or
  1730.       free_space=cargo_capacity(ship)-ship->weight;  // taking heavy armor into account
  1731.  
  1732.    The function also returns a -1 if the unit specified is not a cargo vessel,
  1733.    so it can be used to quickly check that as well.
  1734. */
  1735. int cargo_capacity(ship)
  1736. struct Unit *ship;
  1737. {
  1738.    int capacity;
  1739.  
  1740.    switch (ship->type) {
  1741.       case TRANSPORT:
  1742.          capacity = 6;
  1743.          break;
  1744.       case CARRIER:
  1745.          capacity = 8;
  1746.          break;
  1747.       case AIRBASE:
  1748.          capacity = 20;
  1749.          break;
  1750.       default:
  1751.          return -1;
  1752.    }
  1753.    capacity -= ship->damage*2;   // each hit reduces capacity by two units
  1754.    return capacity;
  1755. }
  1756.  
  1757.  
  1758. /*
  1759.    This function returns the weight (for loading/unloading purposes) of a given
  1760.    type of cargo.  It appears simple to the point of redundancy right now, but it
  1761.    could easily become more complex as the stacking & transporting rules are
  1762.    tinkered with.
  1763. */
  1764.  
  1765. int cargo_weight(type)
  1766. int type;
  1767. {
  1768.    int weight;
  1769.  
  1770.    switch (type) {
  1771.       case RIFLE:
  1772.       case FIGHTER:
  1773.       case BOMBER:
  1774.       case AIRCAV:
  1775.          weight = 1;
  1776.          break;
  1777.       case ARMOR:
  1778.          weight = 2;
  1779.          break;
  1780.       default:
  1781.          weight = 100;  // an impossibly high value
  1782.    }
  1783.    return weight;
  1784. }
  1785.  
  1786.  
  1787. /*
  1788.    When a ship has been given ORDER_LOAD, it should be referred here.  This
  1789.    function will search for loadable cargo units in the same hex (as in a
  1790.    city, for example) or in surrounding hexes and attempt to load them on
  1791.    the ship.     When  the  ship  is  filled,  load_ship()  will  clear  its
  1792.    orders.  Until that happens, however, load_ship() should be called every
  1793.    turn for it.
  1794. */
  1795.  
  1796. void load_ship(ship)
  1797. struct Unit *ship;
  1798. {
  1799.    struct Unit *cargo;
  1800.    int numhexes = adjacent(ship->col,ship->row);
  1801.    int index, access;
  1802.    int units_loaded=0, weight;
  1803.    BOOL valid_cargo;
  1804.  
  1805.    if (cargo_capacity(ship)<0)
  1806.       return;    // not a cargo vessel
  1807.    if (ship->cargo>=cargo_capacity(ship) || ship->weight>=cargo_capacity(ship)) {
  1808.       clear_orders(ship);  // ship already full; clear orders
  1809.       return;
  1810.    }
  1811.  
  1812.    // this gives me an index of all hexes I can load units from
  1813.    hexlist[numhexes++].col = ship->col;
  1814.    hexlist[numhexes++].row = ship->row;
  1815.    cargo = (struct Unit *)unit_list.mlh_Head;
  1816.    for (;cargo->unode.mln_Succ;cargo=(struct Unit *)cargo->unode.mln_Succ)
  1817.       if (unit_readiness(cargo)&UNIT_READY) {
  1818.          // search to see if its in an accessible hex
  1819.          access = FALSE;
  1820.          for (index=numhexes;index>=0;index--)
  1821.             if (cargo->col==hexlist[index].col && cargo->row==hexlist[index].row)
  1822.                access = TRUE;
  1823.          if (access==FALSE)
  1824.             continue;   // too far away; try next unit
  1825.          if (cargo->ship)
  1826.             continue;   // unit is already on a ship!
  1827.          // try to determine whether this is the right kind of cargo
  1828.          if (ship->type==TRANSPORT)
  1829.             valid_cargo = (cargo->type==RIFLE || cargo->type==ARMOR);
  1830.          else
  1831.             valid_cargo = (cargo->type==FIGHTER);  // only fighters in v0.14+
  1832. //            valid_cargo = (wishbook[cargo->type].range>0);
  1833.  
  1834.          // see if armor is too heavy
  1835.          weight = cargo_weight(cargo->type);
  1836.          if (cargo_capacity(ship)-ship->weight < weight)
  1837.             valid_cargo = FALSE;
  1838.  
  1839.          if (valid_cargo) {
  1840.             units_loaded++;
  1841.             clear_orders(cargo);
  1842.             cargo->move = 0;  // loading expends movement
  1843.             cargo->ship = ship;
  1844.             cargo->col = ship->col;
  1845.             cargo->row = ship->row;
  1846.             if (ship->type==TRANSPORT)
  1847.                give_orders(cargo,ORDER_SENTRY,0,0,-1);
  1848.             ship->cargo++;
  1849.             ship->weight += weight;
  1850.  
  1851.             if (ship->cargo>=cargo_capacity(ship) || ship->weight>=cargo_capacity(ship)) {
  1852.                clear_orders(ship);
  1853.                break;
  1854.             FI
  1855.          FI
  1856.       FI
  1857.    if (units_loaded>0) {
  1858.       if (PLAYER.show&SHOW_GRP)
  1859.          explore_at_hex(player,ship->col,ship->row,VISIBLE,FALSE);
  1860.       if (PLAYER.show&SHOW_REQ) {
  1861.          sprintf(foo,"Your %s %s has taken on board %ld units.",
  1862.             wishbook[ship->type].name, ship->name, units_loaded);
  1863.          if (ship->cargo>=cargo_capacity(ship)) {
  1864.             sprintf(bar,"\nThe ship is now fully loaded.");
  1865.             strcat(foo,bar);
  1866.          FI
  1867.          (void)rtEZRequestTags(foo,"Okay",NULL,NULL,
  1868.             RT_Window,        map_window,
  1869.             RT_ReqPos,        REQPOS_CENTERSCR,
  1870.             RT_LockWindow,    TRUE,
  1871.             RTEZ_Flags,       EZREQF_CENTERTEXT,
  1872.             TAG_DONE );
  1873.       FI
  1874.    FI
  1875. }
  1876.  
  1877.  
  1878.  
  1879. /* When a unit has been given the GOTO order it will move
  1880.    towards the hex recorded in it's orders, clear it's
  1881.    orders if it hits any impediment (move result = -1),
  1882.    check when it reaches it's destination, and turn
  1883.    around and head back if it is doing a PATROL order.
  1884.    Of course it may have died moving, so we need to do all
  1885.    checks at the beginning of the routine.
  1886.    */
  1887. void  do_goto( struct Unit* unit )
  1888. {
  1889.     int    result;
  1890.     char   foo[128];
  1891.  
  1892.     if( unit->orders->dest_unit ) {
  1893.         if( !AI3_AssertUnit( unit->orders->dest_unit ) ) {
  1894.               clear_orders( unit );
  1895.               return;
  1896.         }
  1897.         AI5_CalcPath( unit->type, unit->col, unit->row,
  1898.             unit->orders->dest_unit->col,
  1899.             unit->orders->dest_unit->row, AI5_PATH_GOOD );
  1900.        // Check our fuel supply
  1901.        if( (wishbook[unit->type].range > 0) && (unit->fuel < PathLength) ) {
  1902.          clear_orders( unit );
  1903.          return;
  1904.         }
  1905.     }
  1906.     else {
  1907.         AI5_CalcPath( unit->type, unit->col, unit->row,
  1908.             unit->orders->destx, unit->orders->desty, AI5_PATH_GOOD );
  1909.        // Check our fuel supply
  1910.        if( (wishbook[unit->type].range > 0) && (unit->fuel < PathLength) ) {
  1911.          clear_orders( unit );
  1912.          return;
  1913.         }
  1914.     }
  1915.     if( Path[0] == -1 ) {
  1916.         // No path! Or we got there already.
  1917.         clear_orders( unit );
  1918.         return;
  1919.     }
  1920.     else {
  1921.         while( (unit->move > 0) && (unit->orders) && (Path[0] != -1) ) {
  1922.             result = move_unit_dir( unit, Path[0] );
  1923.             // Check if we couldn't move, missed a chance, attacked a unit
  1924.             //   or city, etc.
  1925.             if( result <= -1 ) return;
  1926.             // Else we did move and nothing bad happened, so let's look.
  1927.             if( unit->orders ) {
  1928.                 if( !unit->orders->dest_unit ) {
  1929.                     if( (unit->col == unit->orders->destx) &&
  1930.                         (unit->row == unit->orders->desty) ) {
  1931.                         // We reached a destination
  1932.                         if( unit->orders->etc >= 0 ) {
  1933.                             // Simple GOTO
  1934.                             clear_orders( unit );
  1935.                             return;
  1936.                         }
  1937.                         else {
  1938.                             // Patrol order - swap 'em.
  1939.                             short temp = unit->orders->destx;
  1940.                             unit->orders->destx = unit->orders->orgx;
  1941.                             unit->orders->orgx = temp;
  1942.                             temp = unit->orders->desty;
  1943.                             unit->orders->desty = unit->orders->orgy;
  1944.                             unit->orders->orgy = temp;
  1945.                             //return;
  1946.              // Calculate a new path BACK.
  1947.              AI5_CalcPath( unit->type, unit->col, unit->row,
  1948.                  unit->orders->destx,
  1949.                  unit->orders->desty, AI5_PATH_GOOD );
  1950.                         }
  1951.                     } // End if we arrived
  1952.                   else {
  1953.                       // Create a new path for the next loop
  1954.                       AI5_CalcPath( unit->type, unit->col, unit->row,
  1955.                          unit->orders->destx,
  1956.                          unit->orders->desty, AI5_PATH_GOOD );
  1957.                      // Check our fuel supply
  1958.                      if( (wishbook[unit->type].range > 0) &&
  1959.                          (unit->fuel < PathLength) ) {
  1960.                          clear_orders( unit );
  1961.                          return;
  1962.                      }
  1963.                     // Check our path, see if it got longer
  1964.                     if( unit->orders->etc > 0 ) {
  1965.                         unit->orders->etc--;
  1966.                         if( PathLength - unit->orders->etc > 3 ) {
  1967.                            // we have a major obstacle
  1968.                            if( unit->name ) {
  1969.                                sprintf
  1970.                                    (foo,
  1971.                                    "Major obstacle detected for %s %s\nmoving from %ld to %ld",
  1972.                                    UnitString[unit->type],
  1973.                                    unit->name,
  1974.                                    unit->col, unit->row,
  1975.                                    unit->orders->destx,
  1976.                                    unit->orders->desty);
  1977.                            } // End if unit->name
  1978.                            else {
  1979.                               sprintf
  1980.                                   (foo,
  1981.                                    "Major obstacle detected for %s\nmoving from %ld to %ld",
  1982.                                    UnitString[unit->type],
  1983.                                    unit->col, unit->row,
  1984.                                    unit->orders->destx,
  1985.                                    unit->orders->desty);
  1986.                            } // End else no unit->name
  1987.                            if( rtEZRequestTags
  1988.                                (foo,
  1989.                                 "Clear Orders|Continue Anyway",
  1990.                                 NULL,NULL,
  1991.                                 RT_Window,     map_window,
  1992.                                 RT_ReqPos,     REQPOS_CENTERSCR,
  1993.                                 RT_LockWindow, TRUE,
  1994.                                 RTEZ_Flags,    EZREQF_CENTERTEXT,
  1995.                                 TAG_DONE ) ) {
  1996.                                   clear_orders( unit );
  1997.                             } // End if user wants to clear orders
  1998.                         } // End if major obstacle
  1999.                       } // End if unit->orders->etc >=0
  2000.                     } // End else we have not yet arrived
  2001.                 } // End if not a goto-unit
  2002.               else {
  2003.                   if( AI3_AssertUnit( unit->orders->dest_unit ) ) {
  2004.                       // check for reaching our destination unit
  2005.                       if( (unit->col == unit->orders->dest_unit->col) &&
  2006.                          (unit->row == unit->orders->dest_unit->row) ) {
  2007.                          clear_orders(unit);
  2008.                          return;
  2009.                       }
  2010.                       else {
  2011.                          // Create a new path for the next loop
  2012.                          AI5_CalcPath( unit->type, unit->col, unit->row,
  2013.                              unit->orders->dest_unit->col,
  2014.                              unit->orders->dest_unit->row,
  2015.                              AI5_PATH_GOOD );
  2016.                          // Check our fuel supply
  2017.                          if( (wishbook[unit->type].range > 0) &&
  2018.                              (unit->fuel < PathLength) ) {
  2019.                              clear_orders( unit );
  2020.                              return;
  2021.                          }
  2022.                        // Check our path, see if it got longer
  2023.                      if( unit->orders->etc > 0 ) {
  2024.                            unit->orders->etc--;
  2025.                          if( PathLength - unit->orders->etc > 3 ) {
  2026.                                // we have a major obstacle
  2027.                                if( unit->name ) {
  2028.                                  sprintf
  2029.                                  (foo,
  2030.                                   "Major obstacle detected for %s %s\nmoving from %ld to %ld",
  2031.                                   UnitString[unit->type],
  2032.                                   unit->name,
  2033.                                   unit->col, unit->row,
  2034.                                   unit->orders->destx,
  2035.                                   unit->orders->desty);
  2036.                                } // End if unit->name
  2037.                                else {
  2038.                                  sprintf
  2039.                                   (foo,
  2040.                                   "Major obstacle detected for %s\nmoving from %ld to %ld",
  2041.                                   UnitString[unit->type],
  2042.                                   unit->col, unit->row,
  2043.                                   unit->orders->destx,
  2044.                                   unit->orders->desty);
  2045.                                } // End else no unit->name
  2046.                                if( rtEZRequestTags
  2047.                                   (foo,
  2048.                                   "Clear Orders|Continue Anyway",
  2049.                                   NULL,NULL,
  2050.                                   RT_Window,     map_window,
  2051.                                   RT_ReqPos,     REQPOS_CENTERSCR,
  2052.                                   RT_LockWindow, TRUE,
  2053.                                   RTEZ_Flags,    EZREQF_CENTERTEXT,
  2054.                                    TAG_DONE ) ) {
  2055.                                        clear_orders( unit );
  2056.                                }
  2057.                            } // End if major obstacle
  2058.                             } // end if unit->orders->etc >= 0
  2059.                      } // End else we did not reach our destination unit
  2060.                   }// end if we still HAVE a destination unit
  2061.               } // end else we are trying to reach a unit not a location
  2062.             } // End if we still have orders
  2063.         }// end while
  2064.     } // end else we have a path.
  2065.     return;
  2066. }
  2067.  
  2068. // this is where a unit is moving on the map, checking all the movment
  2069. // rules, possible combat and such as we go
  2070.  
  2071. int move_unit_dir(unit,dir)
  2072. struct Unit *unit;
  2073. enum Direction dir;
  2074. {
  2075.    int targx = unit->col, targy = unit->row;
  2076.  
  2077.    switch (dir) {
  2078.       case EAST:
  2079.          targx++;
  2080.          break;
  2081.       case WEST:
  2082.          targx--;
  2083.          break;
  2084.       case NORTHEAST:
  2085.          targy--;
  2086.          if (unit->row%2) // odd number
  2087.             targx++;
  2088.          break;
  2089.       case NORTHWEST:
  2090.          targy--;
  2091.          if (!(unit->row%2)) // even number
  2092.             targx--;
  2093.          break;
  2094.       case SOUTHEAST:
  2095.          targy++;
  2096.          if (unit->row%2) // odd number
  2097.             targx++;
  2098.          break;
  2099.       case SOUTHWEST:
  2100.          targy++;
  2101.          if (!(unit->row%2)) // even number
  2102.             targx--;
  2103.          break;
  2104.       default:
  2105.          return (-8);
  2106.    }
  2107.    return (move_unit_xy(unit,targx,targy));
  2108. }
  2109.  
  2110.  
  2111. int move_unit_xy(unit,targx,targy)
  2112. struct Unit *unit;
  2113. int targx, targy;
  2114. {
  2115.         //BSH
  2116.         // This routine will now return integer values for results
  2117.         //              -6 - Attack on City (successful - but still lose unit)
  2118.         //              -5 - Attack on City (failed)
  2119.         //              -4 - Attack on enemy unit(s) (failed)
  2120.         //              -3 - Unit ran out of fuel and crashed
  2121.         //              -2 - Unit failed to move this turn (missed chance)
  2122.         //              -1 - Unit cannot move to target hex
  2123.         //              0 - Move occurred
  2124.         //              1 - Moved - Attacked enemy unit(s) (won)
  2125.         //              2 - Moved - Overflew carrier
  2126.         //              3 - Moved - Boarded ship
  2127.         //              4 - Unit moved, but unit(s) on board were lost
  2128.         // To use the unit over again, check for return > -3
  2129.  
  2130.    BOOL cargo_killed=FALSE;
  2131.    int terrain, movement_cost=(-1);
  2132.    struct City *city=city_hereP(targx,targy);  // so I don't call it over and over
  2133.  
  2134.    // correct values for wrap, if it's active
  2135.    if (wrap) {
  2136.       if (targx<0)
  2137.          targx += width;
  2138.       if (targx>=width)
  2139.          targx -= width;
  2140.       if (targy<0)
  2141.          targy += height;
  2142.       if (targy>=height)
  2143.          targy -= height;
  2144.    }
  2145.  
  2146.    /* he certainly can't move off the map! */
  2147.    //  *************  Adding disable for non-users ***************
  2148.    if (targx<0 || targx>=width || targy<0 || targy>=height) {
  2149.       if (PLAYER.show & SHOW_REQ)
  2150.          tell_user2("You cannot move off the map.",FALSE,DONK_SOUND);
  2151.       return (-1);
  2152.    FI
  2153.  
  2154.    /***
  2155.       Now I want to branch out.  I already know the user is trying to move
  2156.       a unit onto another hex.  I need to analyze the contents of that hex
  2157.       and see if he is attacking a city, attacking an enemy unit, boarding
  2158.       a ship, or just maneuvering normally.  I will branch out to various
  2159.       functions for each of these cases, except maneuvering which I can
  2160.       handle here.
  2161.    ***/
  2162.  
  2163.    // check for a destination city
  2164.    if (city)
  2165.       if (city->owner==player)  /* it's his own city */
  2166.          goto normal_movement;
  2167.    else {
  2168.       // if this unit can attack a city
  2169.       attack_hex(unit,targx,targy);  // attacking another city
  2170.       if ((unit->type == ARMOR) || (unit->type == RIFLE) || (unit->type == AIRCAV) || (unit->type == BOMBER)) {
  2171.          if (hex_owner(targx, targy)!=player)
  2172.             return -5;
  2173.          else
  2174.             return -6;
  2175.       } else
  2176.          return -1;
  2177.    }
  2178.  
  2179.    // check for hostile military units to attack
  2180.    {
  2181.       struct Unit *target = (struct Unit *)unit_list.mlh_Head;
  2182.       for (; target->unode.mln_Succ; target=(struct Unit *)target->unode.mln_Succ)
  2183.          if (target->col==targx && target->row==targy && target->owner!=player) {
  2184.             attack_hex(unit,targx,targy);
  2185.             if (hex_owner(targx, targy)!=player)
  2186.                return (-4);
  2187.             else
  2188.                return 1;
  2189.          FI
  2190.    }
  2191.  
  2192.    // check for friendly ship to board, or airbase to land on
  2193.    if (board_ship(unit,targx,targy)) {
  2194.       if (unit->ship==NULL) {    /* i.e. if unit failed to board the ship */
  2195.     /* if it's an aircraft, ask if they want to overfly the hex anyhow */
  2196.          if (wishbook[unit->type].range>0) {
  2197.             int choice;
  2198.             // If we have a human, ask him
  2199.             if (PLAYER.show & SHOW_REQ) {
  2200.                 if(PLAYER.soundfx == SOUND_ALL)
  2201.                      playSound(DONK_SOUND,PLAYER.snd_vol);
  2202.                 choice = rtEZRequestTags(\
  2203.                         "The carrier cannot accept your unit.\nDo you wish to overfly it?",
  2204.                         "Overfly|Abort", NULL, NULL,
  2205.                         RT_Window,        map_window,
  2206.                         RT_ReqPos,        REQPOS_CENTERSCR,
  2207.                         RT_LockWindow,    TRUE,
  2208.                         RT_ShareIDCMP,    TRUE,
  2209.                         RTEZ_Flags,       EZREQF_CENTERTEXT,
  2210.                         TAG_DONE );
  2211.             }
  2212.             // Otherwise, just do the overfly
  2213.                 else  choice = 1;
  2214.             if (choice==0)  return (-1);
  2215.          } else {
  2216.                 if (PLAYER.show & SHOW_REQ)
  2217.                 tell_user2("No ship in that hex can accept your troops.",FALSE,DONK_SOUND);
  2218.             return (-1);
  2219.          FI
  2220.       } else   // unit successfully boarded the ship
  2221.          return (3);
  2222.    FI
  2223.  
  2224.  normal_movement:
  2225.    // check to see if stacking rules allow him to enter this hex
  2226.    if (opt.stacking==FALSE && city==NULL) {
  2227.       BOOL valid = TRUE;
  2228.       struct Unit *target = (struct Unit *)unit_list.mlh_Head;
  2229.  
  2230.       for (; target->unode.mln_Succ; target=(struct Unit *)target->unode.mln_Succ)
  2231.          if (target->col==targx && target->row==targy) {
  2232.             int weight=cargo_weight(unit->type);
  2233.  
  2234.             valid = FALSE;
  2235.  
  2236.             if (unit->type==ARMOR || unit->type==RIFLE)
  2237.                if (target->type==TRANSPORT && target->weight<=(cargo_capacity(target)-weight)) {
  2238.                   valid = TRUE;
  2239.                   break;
  2240.                FI
  2241.             if (wishbook[unit->type].range>0)   // aircraft
  2242.                if (target->type==CARRIER && target->weight<(cargo_capacity(target)-weight)) {
  2243.                   valid = TRUE;
  2244.                   break;
  2245.                FI
  2246.          FI
  2247.       if (valid==FALSE) {   // cannot enter this hex
  2248.          if (PLAYER.show & SHOW_REQ)
  2249.                 tell_user2("Your unit cannot stack in that hex.",FALSE,DONK_SOUND);
  2250.          return (-1);
  2251.       FI
  2252.    FI
  2253.  
  2254.    terrain = get(t_grid,targx,targy);
  2255.    movement_cost = movement_cost_table[unit->type][terrain];
  2256.  
  2257.    if (city)
  2258.       movement_cost = 10;  // always easy to enter a friendly city!
  2259.  
  2260.    // is he on the road?
  2261.    if (unit->type==RIFLE || unit->type==ARMOR)
  2262.       if (get_flags(t_grid,unit->col,unit->row)&get_flags(t_grid,targx,targy)&ROAD)
  2263.          movement_cost = 30;
  2264.  
  2265.    if (terrain==HEX_FORBID) { /* Just in case a road is here! *ELS* */
  2266.       if (PLAYER.show & SHOW_REQ)
  2267.         tell_user2("Your unit cannot traverse that kind of terrain.",FALSE,DONK_SOUND);
  2268.       return (-1);
  2269.    FI
  2270.  
  2271.    if (movement_cost<0) { /* it must be a terrain this unit can't traverse */
  2272.       if (PLAYER.show & SHOW_REQ)
  2273.         tell_user2("Your unit cannot traverse that kind of terrain.",FALSE,DONK_SOUND);
  2274.       return (-1);
  2275.    FI
  2276.  
  2277.    if (city)
  2278.       movement_cost = 10;  // yes, even if there is a road here
  2279.  
  2280.    if (unit->ship)
  2281.       movement_cost = 60;  // this is always true when unloading from a ship
  2282.  
  2283.    /* if a unit's remaining movement points drop to 10 or less (which is 1/6 of
  2284.    ** a "standard" one hex move, then I round it down to zero; this is also the
  2285.    ** minimum for attacking, loading onto ships, entering cities, etc.
  2286.    */
  2287.    if (movement_cost<unit->move) {   // he definitely has enough movement for this
  2288.       unit->move -= movement_cost;
  2289.       if (unit->move<10) {
  2290.          unit->move = 0;
  2291.          control_flag = UNIT_DONE;
  2292.       FI
  2293.    } else {    // he only has enough movement for a /chance/ at this
  2294.       int chance = unit->move;
  2295.  
  2296.       unit->move = 0;
  2297.       control_flag = UNIT_DONE;
  2298.       if (RangeRand(movement_cost)>chance) {
  2299.         //BSH
  2300.          if (PLAYER.soundfx == SOUND_ALL)
  2301.                  playSound(SMASH_SOUND,PLAYER.snd_vol);
  2302.          return (-2);     // his movement attempt failed
  2303.       FI
  2304.    FI
  2305.  
  2306.    // follow him with scrolling, if necessary
  2307.    //BSH
  2308.    if (PLAYER.show & SHOW_GRP) {   // Was SHOW_REQ  *****
  2309.       if (need_to_scrollP(targx,targy)) {
  2310.          int ox = xoffs, oy = yoffs;
  2311.  
  2312.          set_display_offsets(targx,targy);
  2313.          GP_smart_scroll(ox,oy);
  2314.       FI
  2315.       // animated movement
  2316.       anim_move(unit,unit->col,unit->row,targx,targy);
  2317.    FI
  2318.  
  2319.    // move the unit
  2320.    unit->col = targx;
  2321.    unit->row = targy;
  2322.    if (unit->ship) {     // he must be unloading from a ship
  2323.       unit->ship->cargo--;
  2324.       unit->ship->weight -= cargo_weight(unit->type);
  2325.       if (unit->ship->type==TRANSPORT) {
  2326.          unit->move = 0;   // unloading from a transport expends movement
  2327.          control_flag = UNIT_DONE;
  2328.       FI
  2329.       unit->ship=NULL;  // so we unlink him from it
  2330.    FI
  2331.    if (unit->cargo>0) {   // ship moves, cargo moves
  2332.       int ctr, capacity, num_hexes=adjacent(targx,targy);
  2333.       struct Unit *cargo;     BOOL unsentry=FALSE;
  2334.  
  2335.       /*
  2336.          If the ship has been damaged, it may no longer be able to carry all
  2337.          the troops or aircraft on board it.  If that is the case, here is
  2338.          where we kill off the extra ones, when the ship moves.  Of course,
  2339.          they can survive if the ship has just moved into a city!
  2340.       */
  2341.       capacity = 6;
  2342.       if (unit->type==CARRIER)
  2343.          capacity = 8;
  2344.       capacity -= unit->damage*2;
  2345.       /*
  2346.          kill units until capacity == cargo
  2347.          multiple searches may be required because the destruction of units
  2348.          messes up the list structure
  2349.       */
  2350.       ctr = 0;
  2351.       while (unit->cargo>capacity)
  2352.          for (cargo=(struct Unit *)unit_list.mlh_Head;cargo->unode.mln_Succ;cargo=(struct Unit *)cargo->unode.mln_Succ)
  2353.             if (cargo->ship==unit) {
  2354.                unit->cargo--;
  2355.                if (city)   // cargo is merely unloaded into the city
  2356.                   cargo->ship = NULL;
  2357.                else {      // cargo is lost at sea
  2358.                   ctr++;
  2359.                   Remove((struct Node *)cargo);
  2360.                   destruct_unit(cargo);
  2361.                   break;
  2362.                FI
  2363.             FI
  2364.       if (ctr) {
  2365.          cargo_killed = TRUE;
  2366.          if (PLAYER.show & SHOW_REQ) {
  2367.             // I must tell the user the bad news!  Number of units lost == ctr.
  2368.             sprintf(foo,"Because of battle damage, the ship is overloaded.\nYou lost %ld cargo to the sea!",ctr);
  2369.             (void)rtEZRequestTags(foo,
  2370.                "Drat!",NULL,NULL,
  2371.                RT_Window,        map_window,
  2372.                RT_ReqPos,        REQPOS_CENTERSCR,
  2373.                RT_LockWindow,    TRUE,
  2374.                RTEZ_Flags,       EZREQF_CENTERTEXT,
  2375.                TAG_DONE );
  2376.          FI
  2377.       FI
  2378.  
  2379.       /*
  2380.          Find out if there is a hostile unit or city, or neutral city
  2381.          in a hex adjacent to this one; if so, un-sentry RIFLE, ARMOR or AIRCAV.
  2382.          The "unsentry" flag will determine this.
  2383.       */
  2384.       for (ctr=0; ctr<num_hexes; ctr++) {
  2385.          int owner=hex_owner(hexlist[ctr].col,hexlist[ctr].row);
  2386.          struct City *metro=city_hereP(hexlist[ctr].col,hexlist[ctr].row);
  2387.  
  2388.          if (owner>0 && owner!=player)
  2389.             unsentry=TRUE;
  2390.          if (metro)     // check for neutral cities
  2391.             if (metro->owner!=player)
  2392.                unsentry=TRUE;
  2393.       OD
  2394.       for (cargo=(struct Unit *)unit_list.mlh_Head;cargo->unode.mln_Succ;cargo=(struct Unit *)cargo->unode.mln_Succ)
  2395.          if (cargo->ship==unit) {
  2396.             cargo->col=targx;    cargo->row=targy;
  2397.             if (unsentry)
  2398.                switch (cargo->type) {
  2399.                   case RIFLE:
  2400.                   case ARMOR:
  2401.                   case AIRCAV:
  2402.                      clear_orders(cargo);
  2403.                }
  2404.          FI
  2405.    FI
  2406.    // units entering into cities get repaired
  2407.    if (city) {
  2408.       if (unit->damage>0)
  2409.          unit->damage--;
  2410.       unit->move = 0;
  2411.       control_flag = UNIT_DONE;
  2412.    FI
  2413.    if (wishbook[unit->type].range>0) {   // units that need fuel, i.e. aircraft
  2414.       if( unit->ship == NULL ) unit->fuel--; // Added this if statement here,
  2415.       //  was just unit->fuel--, trying to fix bug where aircraft on carrier
  2416.       // lose fuel as carrier moves.
  2417.       if (city) {  // refuel
  2418.          unit->fuel = wishbook[unit->type].range;
  2419.          unit->move = 0;  /* after entering a city, the plane's turn is over */
  2420.          control_flag = UNIT_DONE;
  2421.       FI
  2422.       if (unit->fuel<=0) {
  2423.          // aircraft out of fuel, crashes!
  2424.          if (PLAYER.show & SHOW_REQ) {
  2425.             plot_mapobject(targx,targy,154,30);    // draw explosion
  2426.             context_sound(BOOM_SOUND);              // kaboomy!
  2427.             // now tell the player the awful news
  2428.             sprintf(foo,"Your %s has run out of fuel and crashed!",wishbook[unit->type].name);
  2429.             (void)rtEZRequestTags(foo,"Oh no!",NULL,NULL,
  2430.                RT_Window,     map_window,
  2431.                RT_ReqPos,     REQPOS_CENTERWIN,
  2432.                RT_LockWindow, TRUE,
  2433.                TAG_DONE);
  2434.          FI
  2435.          Remove((struct Node *)unit);           // kill the plane
  2436.          roster[unit->owner].ulc[unit->type]++;
  2437.          control_flag = UNIT_LOST;
  2438.          destruct_unit(unit);
  2439.          if (PLAYER.show & SHOW_GRP)  // Was SHOW_REQ ****
  2440.             explore_at_hex(player,targx,targy,VISIBLE,FALSE);
  2441.          else
  2442.             explore_at_hex(player,targx,targy,INVISIBLE,FALSE);
  2443.          return (-3);
  2444.       FI
  2445.    FI
  2446.    if (PLAYER.show & SHOW_GRP) {  // Was SHOW_REQ ****
  2447.       unit_status_bar(unit);
  2448.       explore_at_hex(player,targx,targy,VISIBLE,FALSE);
  2449.       // Special for AI players showing movement in realtime (cheat mode)
  2450.       if( NONHUMAN( PLAYER.type ) ) {
  2451.           //printf( "Calling GP_update_at_hex\n");
  2452.           GP_update_at_hex( targx, targy );
  2453.       }
  2454.    } else {
  2455.       explore_at_hex(player,targx,targy,INVISIBLE,FALSE);
  2456.    FI
  2457.    if (cargo_killed==TRUE)
  2458.       return (4);
  2459.    return (0);
  2460. }
  2461.  
  2462.  
  2463. void build_move_menu()
  2464. {  // prepare the movement-mode menu for use
  2465.    struct NewMenu new_menu_strip[]  = {
  2466.       { NM_TITLE, "Project", NULL, NM_MENUDISABLED, NULL, NULL },
  2467.       { NM_ITEM, "Save Game", "S", ITEMTEXT, NULL, NULL },
  2468.       { NM_ITEM, "Save As...", NULL, ITEMTEXT, NULL, NULL },
  2469.       { NM_ITEM, NM_BARLABEL, NULL, ITEMTEXT, NULL, NULL },
  2470.       { NM_ITEM, "Exit Game", "X", ITEMTEXT, NULL, NULL },
  2471.       { NM_ITEM, "Quit Program", "Q", ITEMTEXT, NULL, NULL },
  2472.       { NM_TITLE, "Reports", NULL, NM_MENUDISABLED, NULL, NULL },
  2473.       { NM_ITEM, "World Map", NULL, ITEMTEXT, NULL, NULL },
  2474.       { NM_ITEM, "Status", NULL, ITEMTEXT, NULL, NULL },
  2475.       { NM_ITEM, "Combat Report", NULL, ITEMTEXT, NULL, NULL },
  2476.       { NM_ITEM, "Unit Info", NULL, ITEMTEXT, NULL, NULL },
  2477.       { NM_ITEM, "Production Map", NULL, ITEMTEXT, NULL, NULL },
  2478.       { NM_ITEM, "Ship Report", NULL, ITEMTEXT, NULL, NULL },
  2479.       { NM_ITEM, "Advisor's Recommendation", "Z", ITEMTEXT, NULL, NULL },
  2480.       { NM_TITLE, "Orders", NULL, NM_MENUDISABLED, NULL, NULL },
  2481.       { NM_ITEM, "Clear Orders     O", NULL, ITEMTEXT, NULL, NULL },
  2482.       { NM_ITEM, "Sentry           S", NULL, ITEMTEXT, NULL, NULL },
  2483.       { NM_ITEM, "Build Airbase    A", NULL, ITEMTEXT, NULL, NULL },
  2484.       { NM_ITEM, "Lay/Sweep Mines  M", NULL, ITEMTEXT, NULL, NULL },
  2485.       { NM_ITEM, "Go Home          H", NULL, ITEMTEXT, NULL, NULL },
  2486.       { NM_ITEM, "Go Direction     *", NULL, ITEMTEXT, NULL, NULL },
  2487.       { NM_ITEM, "Go Random        *", NULL, ITEMTEXT, NULL, NULL },
  2488.       { NM_ITEM, "Move To          *", NULL, ITEMTEXT, NULL, NULL },
  2489.       { NM_ITEM, "Patrol To        *", NULL, ITEMTEXT, NULL, NULL },
  2490.       { NM_ITEM, "Escort Ship      *", NULL, ITEMTEXT, NULL, NULL },
  2491.       { NM_ITEM, "Load Ship        L", NULL, ITEMTEXT, NULL, NULL },
  2492.       { NM_ITEM, "Unload Ship      U", NULL, ITEMTEXT, NULL, NULL },
  2493.       { NM_ITEM, "Skip Move    [SPC]", NULL, ITEMTEXT, NULL, NULL },
  2494.       { NM_TITLE, "Commands", NULL, NM_MENUDISABLED, NULL, NULL },
  2495.       { NM_ITEM, "Survey Mode    V", NULL, ITEMTEXT, NULL, NULL },
  2496.       { NM_ITEM, "Wait           W", NULL, ITEMTEXT, NULL, NULL },
  2497.       { NM_ITEM, "Cycle Stack    5", NULL, ITEMTEXT, NULL, NULL },
  2498.       { NM_ITEM, "Flight Paths   *", NULL, ITEMTEXT, NULL, NULL },
  2499.       { NM_ITEM, "Center Screen  C", NULL, ITEMTEXT, NULL, NULL },
  2500.       { NM_TITLE, "Other", NULL, NM_MENUDISABLED, NULL, NULL },
  2501.       { NM_ITEM, "Prefs", NULL, ITEMTEXT, NULL, NULL },
  2502.       { NM_ITEM, "Commanders", NULL, ITEMTEXT, NULL, NULL },
  2503.       { NM_ITEM, "View", NULL, ITEMTEXT, NULL, NULL },
  2504.       { NM_END, NULL, NULL, NULL, NULL, NULL }
  2505.    };
  2506.    if (!(move_menu_strip = CreateMenus(new_menu_strip,GTMN_FrontPen,BLACK,TAG_END)))
  2507.       clean_exit(1,"ERROR: Unable to create menu strip for movement mode!");
  2508.    if (!(LayoutMenus(move_menu_strip,vi,TAG_END)))
  2509.       clean_exit(1,"ERROR: Unable to layout movement menus!");
  2510. }
  2511.  
  2512.  
  2513. void build_survey_menu()
  2514. {  // prepare the survey-mode menu for use
  2515.    struct NewMenu new_menu_strip[]  = {
  2516.       { NM_TITLE, "Project", NULL, NM_MENUDISABLED, NULL, NULL },
  2517.       { NM_ITEM, "Save Game", "S", ITEMTEXT, NULL, NULL },
  2518.       { NM_ITEM, "Save As...", NULL, ITEMTEXT, NULL, NULL },
  2519.       { NM_ITEM, NM_BARLABEL, NULL, ITEMTEXT, NULL, NULL },
  2520.       { NM_ITEM, "Exit Game", "X", ITEMTEXT, NULL, NULL },
  2521.       { NM_ITEM, "Quit Program", "Q", ITEMTEXT, NULL, NULL },
  2522.       { NM_TITLE, "Reports", NULL, NM_MENUDISABLED, NULL, NULL },
  2523.       { NM_ITEM, "World Map", NULL, ITEMTEXT, NULL, NULL },
  2524.       { NM_ITEM, "Status", NULL, ITEMTEXT, NULL, NULL },
  2525.       { NM_ITEM, "Combat Report", NULL, ITEMTEXT, NULL, NULL },
  2526.       { NM_ITEM, "Hex Info", NULL, ITEMTEXT, NULL, NULL },
  2527.       { NM_ITEM, "Production Map", NULL, ITEMTEXT, NULL, NULL },
  2528.       { NM_ITEM, "Ship Report", NULL, ITEMTEXT, NULL, NULL },
  2529.       { NM_TITLE, "Orders", NULL, NM_MENUDISABLED, NULL, NULL },
  2530.       { NM_ITEM, "Clear Orders  O", NULL, ITEMTEXT, NULL, NULL },
  2531.       { NM_ITEM, "Cycle Stack   5", NULL, ITEMTEXT, NULL, NULL },
  2532.       { NM_ITEM, "Activate      A", NULL, ITEMTEXT, NULL, NULL },
  2533.       { NM_TITLE, "Commands", NULL, NM_MENUDISABLED, NULL, NULL },
  2534.       { NM_ITEM, "Move Mode      M", NULL, ITEMTEXT, NULL, NULL },
  2535.       { NM_ITEM, "Group Survey   *", NULL, ITEMTEXT, NULL, NULL },
  2536.       { NM_ITEM, "Flight Paths   *", NULL, ITEMTEXT, NULL, NULL },
  2537.       { NM_ITEM, "Center Screen  C", NULL, ITEMTEXT, NULL, NULL },
  2538.       { NM_TITLE, "Other", NULL, NM_MENUDISABLED, NULL, NULL },
  2539.       { NM_ITEM, "Prefs", NULL, ITEMTEXT, NULL, NULL },
  2540.       { NM_ITEM, "Commanders", NULL, ITEMTEXT, NULL, NULL },
  2541.       { NM_ITEM, "View", NULL, ITEMTEXT, NULL, NULL },
  2542.       { NM_END, NULL, NULL, NULL, NULL, NULL }
  2543.    };
  2544.    if (!(vey_menu_strip = CreateMenus(new_menu_strip,GTMN_FrontPen,BLACK,TAG_END)))
  2545.       clean_exit(1,"ERROR: Unable to create menu strip for survey mode!");
  2546.    if (!(LayoutMenus(vey_menu_strip,vi,TAG_END)))
  2547.       clean_exit(1,"ERROR: Unable to layout survey menus!");
  2548. }
  2549.  
  2550.  
  2551. void build_production_menu()
  2552. {  // prepare the production-mode menu for use
  2553.    struct NewMenu new_menu_strip[]  = {
  2554.       { NM_TITLE, "Project", NULL, NM_MENUDISABLED, NULL, NULL },
  2555.       { NM_ITEM, "Save Game", "S", ITEMTEXT, NULL, NULL },
  2556.       { NM_ITEM, "Save As...", NULL, ITEMTEXT, NULL, NULL },
  2557.       { NM_ITEM, NM_BARLABEL, NULL, ITEMTEXT, NULL, NULL },
  2558.       { NM_ITEM, "Exit Game", "X", ITEMTEXT, NULL, NULL },
  2559.       { NM_ITEM, "Quit Program", "Q", ITEMTEXT, NULL, NULL },
  2560.       { NM_TITLE, "Reports", NULL, NM_MENUDISABLED, NULL, NULL },
  2561.       { NM_ITEM, "World Map", NULL, ITEMTEXT, NULL, NULL },
  2562.       { NM_ITEM, "Status", NULL, ITEMTEXT, NULL, NULL },
  2563.       { NM_ITEM, "Ship Report", NULL, ITEMTEXT, NULL, NULL },
  2564.       { NM_TITLE, "Commands", NULL, NM_MENUDISABLED, NULL, NULL },
  2565.       { NM_ITEM, "Examine City [Enter]", NULL, ITEMTEXT, NULL, NULL },
  2566.       { NM_ITEM, "Move Mode      M", NULL, ITEMTEXT, NULL, NULL },
  2567.       { NM_ITEM, "Survey Mode        V", NULL, ITEMTEXT, NULL, NULL },
  2568.       { NM_ITEM, "Center Screen  C", NULL, ITEMTEXT, NULL, NULL },
  2569.       { NM_TITLE, "Other", NULL, NM_MENUDISABLED, NULL, NULL },
  2570.       { NM_ITEM, "Prefs", NULL, ITEMTEXT, NULL, NULL },
  2571.       { NM_ITEM, "Commanders", NULL, ITEMTEXT, NULL, NULL },
  2572.       { NM_ITEM, "View", NULL, ITEMTEXT, NULL, NULL },
  2573.       { NM_END, NULL, NULL, NULL, NULL, NULL }
  2574.    };
  2575.    if (!(prod_menu_strip = CreateMenus(new_menu_strip,GTMN_FrontPen,BLACK,TAG_END)))
  2576.       clean_exit(1,"ERROR: Unable to create menu strip for production mode!");
  2577.    if (!(LayoutMenus(prod_menu_strip,vi,TAG_END)))
  2578.       clean_exit(1,"ERROR: Unable to layout production menus!");
  2579. }
  2580.  
  2581.  
  2582. /*
  2583.    Give orders to a unit.  Note that some of these values (namely destx, desty
  2584.    and etc) may be set by the calling function or may be set by give_orders(),
  2585.    depending on the specifics of the order being given.
  2586.    This function does NOT actually process the order, but leaves that to the
  2587.    order manager to take care of later.
  2588. */
  2589.  
  2590. void give_orders(unit,token,destx,desty,etc)
  2591. struct Unit *unit;
  2592. int token, destx, desty, etc;
  2593. {
  2594.    struct Order *order=AllocVec((long)sizeof(*order),MEMF_CLEAR);
  2595.    struct City* metro = (struct City *)city_list.mlh_Head;
  2596.    struct City* closest = NULL;
  2597.    struct Unit* lookunit = (struct Unit *)unit_list.mlh_Head;
  2598.    struct Unit* nearest = NULL;
  2599.    char   askstring[80];
  2600.  
  2601.    clear_orders(unit);
  2602.    if (order) {
  2603.       unit->orders = order;
  2604.       order->type = token;
  2605.       order->orgx = unit->col;
  2606.       order->orgy = unit->row;
  2607.       order->processed = FALSE;
  2608.       switch (token) {
  2609.          case ORDER_SENTRY:
  2610.          case ORDER_LOAD:
  2611.             order->destx = unit->col;
  2612.             order->desty = unit->row;
  2613.             order->etc = -1;
  2614.             break;
  2615.          case ORDER_AIRBASE:
  2616.             order->processed = TRUE;
  2617.             break;
  2618.          case ORDER_GOTO:
  2619.             // Check for an aircraft and a possible target carrier
  2620.           // Only FIGHTERS can land on carriers now.
  2621.           if( (unit->type == FIGHTER) //||
  2622.               //(unit->type == BOMBER) ||
  2623.              //(unit->type == AIRCAV)
  2624.               ) {
  2625.                for (; lookunit->unode.mln_Succ; lookunit =
  2626.                  (struct Unit *)lookunit->unode.mln_Succ)
  2627.                     if( (lookunit->owner==player) &&
  2628.                          (lookunit->type == CARRIER) &&
  2629.                         (lookunit->col == destx) &&
  2630.                         (lookunit->row == desty) ) {
  2631.                         // Ask user if he wants it to land on the carrier
  2632.                         sprintf( askstring, "Land aircraft on Carrier %s?",
  2633.                             lookunit->name);
  2634.                         if (rtEZRequestTags(askstring,"Yes|No",
  2635.                             NULL,NULL, RTEZ_Flags,EZREQF_CENTERTEXT,
  2636.                             RT_Window,map_window,RT_ReqPos,
  2637.                             REQPOS_CENTERWIN,RT_LockWindow,TRUE,TAG_END )) {
  2638.                                 order->dest_unit = lookunit;
  2639.                                 order->etc = etc;
  2640.                                 break;
  2641.                         }
  2642.                     }
  2643.                 // End for loop
  2644.             }
  2645.             order->destx = destx;
  2646.             order->desty = desty;
  2647.             order->etc = etc;
  2648.             break;
  2649.          case ORDER_PATROL:
  2650.             order->destx = destx;
  2651.             order->desty = desty;
  2652.             order->etc = -etc;
  2653.             order->type = ORDER_GOTO; /* Important - no token for PATROL */
  2654.             break;
  2655.          case ORDER_HOME:
  2656.            /* Need to find the closest city or carrier we can land on */
  2657.             for ( ; metro->cnode.mln_Succ; metro =
  2658.                 (struct City *)metro->cnode.mln_Succ)
  2659.                 if(metro->owner==player) {
  2660.                   if( !closest )  closest = metro;
  2661.                   else {
  2662.                  if( AI5_GetDist ( unit->col, unit->row,
  2663.                             metro->col, metro->row ) < AI5_GetDist
  2664.                             (unit->col, unit->row, closest->col,
  2665.                             closest->row) )  closest = metro;
  2666.                   }
  2667.               }
  2668.              /* End for */
  2669.                /* Verify the vs. our fuel if applicable, then look for a
  2670.                 carrier. */
  2671.                 if( (unit->type == FIGHTER) || (unit->type == BOMBER) ||
  2672.                   (unit->type == AIRCAV) ) {
  2673.                   /* if the city is too far away, forget it */
  2674.                   if( (closest) &&( AI5_GetDist
  2675.                       (unit->col, unit->row, closest->col, closest->row) >
  2676.                       unit->fuel ) )   closest = NULL;
  2677.                      /* Look for the closest carrier */
  2678.                     for (; lookunit->unode.mln_Succ; lookunit =
  2679.                         (struct Unit *)lookunit->unode.mln_Succ)
  2680.                         if( (lookunit->owner==player) &&
  2681.                            (lookunit->type == CARRIER) ) {
  2682.                         if( !nearest )  nearest = lookunit;
  2683.                         else {
  2684.                             if( AI5_GetDist ( unit->col, unit->row,
  2685.                                     lookunit->col, lookunit->row ) <
  2686.                                     AI5_GetDist (unit->col, unit->row,
  2687.                                     nearest->col, nearest->row) )
  2688.                                     nearest = lookunit;
  2689.                            }
  2690.                       }
  2691.                      /* End for */
  2692.                      /* Do we have a carrier? If so, is it in range? If not,
  2693.                         forget it. */
  2694.                      if( (nearest) && (AI5_GetDist
  2695.                          (unit->col, unit->row, nearest->col, nearest->row) >
  2696.                          unit->fuel ) )    nearest = NULL;
  2697.                   /* Any place to land? */
  2698.                   if( (nearest == NULL) && (closest == NULL) ) {
  2699.                       clear_orders( unit );
  2700.                       break;
  2701.                   }
  2702.                   /* Now, do we have a carrier that is closer? If not, forget
  2703.                      it and go with the city. */
  2704.                   if( (nearest ) && (closest) ) {
  2705.                       if( AI5_GetDist
  2706.                      (unit->col, unit->row, nearest->col, nearest->row) >=
  2707.                      AI5_GetDist
  2708.                      (unit->col, unit->row, closest->col, closest->row) )
  2709.                         nearest = NULL;
  2710.                   }
  2711.                }
  2712.                /* Check them for fuel vs. aircraft - no sense heading for
  2713.                   someplace we can't reach. */
  2714.                 //order->etc = etc;
  2715.                 order->type = ORDER_GOTO;  /* Important - no token for HOME */
  2716.                /* Go for the carrier first */
  2717.                if( nearest ) {
  2718.                   order->dest_unit = nearest;
  2719.                   order->etc = AI5_GetDist(unit->col, unit->row,
  2720.                     order->dest_unit->col, order->dest_unit->row);
  2721.                }
  2722.                else {
  2723.                   if( closest) {
  2724.                       order->destx = closest->col;
  2725.                       order->desty = closest->row;
  2726.                       order->etc = AI5_GetDist(unit->col, unit->row,
  2727.                         order->destx, order->desty);
  2728.                   }
  2729.                   else {
  2730.                       /* Nowhere to go */
  2731.                       clear_orders( unit );
  2732.                   }
  2733.                }
  2734.                break;
  2735.          }
  2736.    }
  2737. }
  2738.  
  2739.  
  2740. /*
  2741.    the purpose of shuffle_units() is to shuffle the current unit in a hex
  2742.    to the bottom of the pile, and activate the next valid one beneath it
  2743.    parameter is the current top unit; return value is the new top unit
  2744. */
  2745. struct Unit *shuffle_units(topunit, survey)
  2746. struct Unit *topunit;
  2747. BOOL survey;   // indicates whether we are in survey mode
  2748. {
  2749.    struct Unit *newunit = (struct Unit *)unit_list.mlh_Head;
  2750.    int col=topunit->col, row=topunit->row;
  2751.  
  2752.    for (; newunit->unode.mln_Succ; newunit=(struct Unit *)newunit->unode.mln_Succ)
  2753.       if (newunit->col==col && newunit->row==row && newunit!=topunit)
  2754.          if (Bool(unit_readiness(newunit)&UNIT_READY) || survey) {
  2755.             // move this unit to the beginning of list
  2756.             Remove((struct Node *)newunit);
  2757.             AddHead((struct List *)&unit_list,(struct Node *)newunit);
  2758.             // move the old unit to the end of the list
  2759.             Remove((struct Node *)topunit);
  2760.             AddTail((struct List *)&unit_list,(struct Node *)topunit);
  2761.             return newunit;
  2762.          FI
  2763.    return topunit;
  2764. }
  2765.  
  2766.  
  2767.  
  2768. void start_blinking_unit(unit)
  2769. struct Unit *unit;
  2770. {  // display the unit to be moved, set up blinking graphics
  2771.    int col = unit->col, row = unit->row;
  2772.    struct City *metro = (struct City *)city_list.mlh_Head;
  2773.    BOOL stacked=(count_units_at(unit->col,unit->row)>1);
  2774.    int token;
  2775.  
  2776.    // draw in the background terrain
  2777.    plot_hex(col,row,get(t_grid,col,row));
  2778.  
  2779.    if (get_flags(PLAYER.map,col,row)&ROAD)
  2780.       GP_draw_roads(col,row);
  2781.  
  2782.    // if there is a city here, draw it
  2783.    for ( ; metro->cnode.mln_Succ; metro = (struct City*)metro->cnode.mln_Succ)
  2784.       if (metro->col==col && metro->row==row)
  2785.          plot_city(col,row,roster[metro->owner].color,FALSE);
  2786.  
  2787.    // if the unit is on board a ship, draw the ship as background
  2788.    if (unit->ship) {
  2789.       token = ORDER_NONE;
  2790.       if (unit->ship->orders)
  2791.          token=unit->ship->orders->type;
  2792.       plot_icon(unit->ship->type,col,row,roster[unit->owner].color,token,TRUE);
  2793.    FI
  2794.  
  2795.    // store this background in a safe place (buffer 0)
  2796.    save_hex_graphics(col,row,0);
  2797.  
  2798.    // plot the unit icon
  2799.    token = ORDER_NONE;
  2800.    if (unit->orders)
  2801.       token = unit->orders->type;
  2802.    plot_icon(unit->type,col,row,roster[unit->owner].color,token,stacked);
  2803.  
  2804.    // just make sure the map icon matches what I'm displaying
  2805.    // otherwise things like order tokens might disappear or be wrong
  2806.    explore_hex(unit->owner,unit->col,unit->row,INVISIBLE,TRUE);
  2807. }
  2808.  
  2809.  
  2810. void movement_mode(current_unit)
  2811. struct Unit **current_unit;
  2812. {
  2813.    struct IntuiMessage *message; // the message the IDCMP sends us
  2814.    struct Unit *unit = *current_unit;
  2815.    unsigned int ticks = 1;
  2816.    BOOL blink_on = TRUE;
  2817.    BOOL stacked;
  2818.  
  2819.    // useful for interpreting IDCMP messages
  2820.    UWORD code;
  2821.    ULONG class;
  2822.    APTR object;
  2823.  
  2824.    if (unit)
  2825.       stacked = (count_units_at(unit->col,unit->row)>1);
  2826.  
  2827.    /*
  2828.       If there is no map display, this is the right time to create one.
  2829.       but first set COL and ROW to default values, searching for a city
  2830.       belonging to the player...
  2831.    */
  2832.    if (unit==NULL && display==FALSE) {
  2833.       struct City *metro=(struct City *)city_list.mlh_Head;
  2834.       for (; metro->cnode.mln_Succ; metro=(struct City *)metro->cnode.mln_Succ)
  2835.          if (metro->owner==player) {
  2836.             cursx=metro->col;   cursy=metro->row;
  2837.             create_player_display(cursx,cursy);
  2838.             break;
  2839.          FI
  2840.    } else if (unit)
  2841.       create_player_display(unit->col,unit->row);
  2842.  
  2843.    /*
  2844.       If the value "unit" passed to movement_mode() is NULL, this signifies that
  2845.       the player has no more units left to move.  So, we must ask him if he wants
  2846.       to continue, then dump him back to the mode manager.  If he wants to
  2847.       continue, we set GO_SURVEY so he is sent to survey mode.  Otherwise we
  2848.       set END_TURN so the manager knows to exit out.
  2849.    */
  2850.    if (unit==NULL) {
  2851.       control_flag = (rtEZRequestTags(\
  2852.          "All your units have moved.\nDo you wish to continue?",
  2853.          " End Turn |Review Map", NULL, NULL,
  2854.          RT_Window,        map_window,
  2855.          RT_ReqPos,        REQPOS_CENTERSCR,
  2856.          RT_LockWindow,    TRUE,
  2857.          RT_ShareIDCMP,    TRUE,
  2858.          RTEZ_Flags,       EZREQF_CENTERTEXT,
  2859.          TAG_DONE
  2860.          )?END_TURN:GO_SURVEY);
  2861.       return;
  2862.    FI
  2863.  
  2864.    if (display==FALSE) {
  2865.       set_display_offsets(unit->col,unit->row);
  2866.       GP_draw_map();
  2867.    } else if (need_to_scrollP(unit->col,unit->row)) {
  2868.       int ox=xoffs, oy=yoffs;
  2869.  
  2870.       set_display_offsets(unit->col,unit->row);
  2871.       GP_smart_scroll(ox,oy);
  2872.    FI
  2873.  
  2874.    // clear orders
  2875.    if (unit->orders) {
  2876.       clear_orders(unit);
  2877.       GP_update_hex_display(unit->col,unit->row);
  2878.    FI
  2879.  
  2880.    start_blinking_unit(unit);
  2881.    unit_status_bar(unit);
  2882.  
  2883.    // activate IDCMP event input
  2884.    ModifyIDCMP(map_window,IDCMP_PLAYGAME);
  2885.  
  2886.    // attach the menu to my window
  2887.    SetMenuStrip(map_window,move_menu_strip);
  2888.  
  2889.    // Enable menus
  2890.    OnMenu(map_window,FULLMENUNUM(0,NOITEM,0));
  2891.    OnMenu(map_window,FULLMENUNUM(1,NOITEM,0));
  2892.    OnMenu(map_window,FULLMENUNUM(2,NOITEM,0));
  2893.    OnMenu(map_window,FULLMENUNUM(3,NOITEM,0));
  2894.    OnMenu(map_window,FULLMENUNUM(4,NOITEM,0));
  2895.  
  2896.    control_flag = 0;    // flag lets me know when unit is done
  2897.  
  2898.    /*
  2899.       This is the most important user input loop of the whole game, where
  2900.       I get the movement of the current unit, let the user scroll around,
  2901.       select various options from the drop-down menus, etc.
  2902.    */
  2903.  
  2904.    while (TRUE) {
  2905.       WaitPort(map_window->UserPort);
  2906.       while (message = GT_GetIMsg(map_window->UserPort)) {
  2907.          code = message->Code;  // MENUNUM
  2908.          object = message->IAddress;  // Gadget
  2909.          class = message->Class;
  2910.          GT_ReplyIMsg(message);
  2911.          if ( class == MENUPICK ) { // MenuItems
  2912.             ClearMenuStrip(map_window);
  2913.             switch (MENUNUM(code)) {
  2914.                case 0:  // Project menu
  2915.                   switch (ITEMNUM(code)) {
  2916.                      case 0:  // save game
  2917.                         {
  2918.                            char pan[216];
  2919.                            build_pan(pan,game_filepath,game_filename);
  2920.                            save_game(pan);
  2921.                         }
  2922.                         break;
  2923.                      case 1:  // save as...
  2924.                         (void)rt_loadsave_game(TRUE);
  2925.                         break;
  2926.                      case 3:  // exit game
  2927.                         if (alert(map_window,"Exit this game.","Are you sure you want to abandon this game now?","Exit|Cancel")) {
  2928.                            control_flag = EXIT_GAME;
  2929.                            return;
  2930.                         FI
  2931.                         break;
  2932.                      case 4:  // quit program
  2933.                         if (alert(map_window,"Exit Invasion Force.","You have a game in progress.\nAre you sure you want to leave Invasion Force now?","Exit|Cancel"))
  2934.                            clean_exit(0,NULL);
  2935.                         break;
  2936.                      default:
  2937.                         (void)rtEZRequestTags("That function is not yot implemented.","Drat!",NULL,NULL,
  2938.                            RT_DEFAULT,TAG_END);
  2939.                   }
  2940.                   break;
  2941.                case 1:  // Reports menu
  2942.                   switch (ITEMNUM(code)) {
  2943.                      case 0:  // World Map
  2944.                         GP_world_view();
  2945.                         break;
  2946.                      case 1:  // Status report
  2947.                         status_report(player);
  2948.                         break;
  2949.                      case 2:  // Combat Report
  2950.                         restore_hex_graphics(unit->col,unit->row,0);
  2951.                         show_combat_report(FALSE);
  2952.                         if (need_to_scrollP(unit->col,unit->row)) {
  2953.                            int ox=xoffs, oy=yoffs;
  2954.  
  2955.                            set_display_offsets(unit->col,unit->row);
  2956.                            GP_smart_scroll(ox,oy);
  2957.                         FI
  2958.                         start_blinking_unit(unit);
  2959.                         break;
  2960.                      case 4:  // Production Map
  2961.                         control_flag = GO_PRODUCTION;
  2962.                         break;
  2963.                     case 6: // Advisor's Recommendation
  2964.                        class = IDCMP_VANILLAKEY;
  2965.                        code = 'z';
  2966.                        break;
  2967.                   }
  2968.                   break;
  2969.                case 2:  // Orders menu
  2970.                   switch (ITEMNUM(code)) {
  2971.                      case 0:  // Clear Orders
  2972.                         class = IDCMP_VANILLAKEY;
  2973.                         code = 'o';
  2974.                         break;
  2975.                      case 1:  // Sentry
  2976.                         class = IDCMP_VANILLAKEY;
  2977.                         code = 's';
  2978.                         break;
  2979.                      case 2:  // Build Airbase
  2980.                         class = IDCMP_VANILLAKEY;
  2981.                         code = 'a';
  2982.                         break;
  2983. //                   case 3:  // Lay/Sweep Mines
  2984.                      case 4:  // Home
  2985.                         class = IDCMP_VANILLAKEY;
  2986.                         code = 'h';
  2987.                         break;
  2988.                      case 10: // Load Ship
  2989.                         class = IDCMP_VANILLAKEY;
  2990.                         code = 'l';
  2991.                         break;
  2992.                      case 11: // Unload Ship
  2993.                         class = IDCMP_VANILLAKEY;
  2994.                         code = 'u';
  2995.                         break;
  2996.                      case 12: // Skip Move
  2997.                         class = IDCMP_VANILLAKEY;
  2998.                         code = ' ';
  2999.                         break;
  3000.                      default:
  3001.                         (void)rtEZRequestTags("That function is not yot implemented.","Drat!",NULL,NULL,
  3002.                            RT_DEFAULT,TAG_END);
  3003.                   }
  3004.                   break;
  3005.                case 3:  // Commands Menu
  3006.                   switch (ITEMNUM(code)) {
  3007.                      case 0:  // Survey Mode
  3008.                         class = IDCMP_VANILLAKEY;
  3009.                         code = 'v';
  3010.                         break;
  3011.                      case 1:  // Wait
  3012.                         class = IDCMP_VANILLAKEY;
  3013.                      code = 'w';
  3014.                      break;
  3015.                      case 2:  // Cycle Stack
  3016.                         class = IDCMP_VANILLAKEY;
  3017.                         code = '5';
  3018.                         break;
  3019.                      case 4:  // Center Screen
  3020.                         class = IDCMP_VANILLAKEY;
  3021.                         code = 'c';
  3022.                         break;
  3023.                   }
  3024.                   break;
  3025.                case 4:  // Other Menu
  3026.                   if (ITEMNUM(code)==0)
  3027.                      player_preferences();
  3028.             };
  3029.          FI
  3030.          // following are key commands available in movement mode
  3031.          if (class==IDCMP_VANILLAKEY) {
  3032.             switch (code) {
  3033.                int direction;
  3034.  
  3035.                case 'a':
  3036.                   { // turn infantry into airbase
  3037.                      build_airbase(unit);
  3038.                      start_blinking_unit(unit);  // so icon in buffer is updated
  3039.                      if (unit->orders)    // i.e. if build_airbase() worked
  3040.                         control_flag = UNIT_DONE;
  3041.                      break;
  3042.                   }
  3043.                case 'q':
  3044.                   { // Special function to turn on and off AI information
  3045.                      AIDataFlag++;
  3046.                      if( AIDataFlag > AI_DATA_FLAG_LIMIT )  {
  3047.                         AIDataFlag = 0;
  3048.                      }
  3049.                      sprintf( foo, "AI Information at level %ld", AIDataFlag );
  3050.                      tell_user2( foo, FALSE, DONK_SOUND );
  3051.                      break;
  3052.                   }
  3053.                case 'z':
  3054.                   {  // New Military Advisor function
  3055.                      int response1 = 0, response2 = 0;
  3056.                      int recommend = AI5_recommend_action( unit, 40, 60 );
  3057.  
  3058.                      if( (recommend >= 0) && (recommend <= 5) ) {
  3059.                         sprintf( foo, "Advisor recommends a move %s",
  3060.                              DirString[recommend] );
  3061.                         response1 = rtEZRequestTags
  3062.                            (foo,"Do it|Show Details|Cancel",
  3063.                            NULL,NULL,
  3064.                            RT_Window,        map_window,
  3065.                            RT_ReqPos,        REQPOS_CENTERSCR,
  3066.                            RT_LockWindow,    TRUE,
  3067.                            RTEZ_Flags,       EZREQF_CENTERTEXT,
  3068.                            TAG_DONE );
  3069.                          // End if we have a recommendation
  3070.                      } else {
  3071.                         response2 = rtEZRequestTags
  3072.                         ("The Advisor has no recommended move for that unit.",
  3073.                            "Show Details|OK",NULL,NULL, RT_DEFAULT,TAG_END);
  3074.                      }
  3075.                      if ( response1 == 1) {
  3076.                         // Taking the advice.  Let's do it then
  3077.                         direction = recommend;
  3078.                         goto mm_moving;
  3079.                      }
  3080.                      if ( (response1 == 2) || (response2 == 1) ) {
  3081.                         // Go through each hex in turn and give him raw data
  3082.                         int index;
  3083.                         struct HexReport rep;
  3084.  
  3085.                         for( index=0; index<6; index++ ) {
  3086.                            /* Fill in the raw data */
  3087.                            short targx, targy;
  3088.                            if (AI1_calc_dir (index, unit->col, unit->row, &targx, &targy) != -1) {
  3089.                               AI5_evaluate_hex( unit, targx, targy, &rep );
  3090.                            }
  3091.                            if( rep.CanAttack || rep.AtRisk ) {
  3092.                               if( unit->name ) {
  3093.                                  sprintf( foo,
  3094.  "%s of %s %s\nAre %ld enemy unit(s) valued at %ld\nThe value of your %ld unit(s) is %ld\nThe odds of winning are %ld  %%",
  3095.                                     DirString[index], unit->name,
  3096.                                     wishbook[unit->type].name,
  3097.                                     rep.TotalEUnits, rep.Gain,
  3098.                                     rep.TotalMyUnits, rep.Risk,
  3099.                                     rep.Odds );
  3100.                                  (void) rtEZRequestTags
  3101.                                       (foo,"OK",
  3102.                                       NULL,NULL,
  3103.                                       RT_Window,      map_window,
  3104.                                       RT_ReqPos,      REQPOS_CENTERSCR,
  3105.                                       RT_LockWindow,  TRUE,
  3106.                                       RTEZ_Flags,     EZREQF_CENTERTEXT,
  3107.                                       TAG_DONE );
  3108.                               } else {
  3109.                                  sprintf( foo,
  3110.  "%s of your %s\nAre %ld enemy unit(s) valued at %ld\nThe value of your %ld unit(s) is %ld\nThe odds of winning are %ld %%",
  3111.                                     DirString[index],
  3112.                                     wishbook[unit->type].name,
  3113.                                     rep.TotalEUnits, rep.Gain,
  3114.                                     rep.TotalMyUnits, rep.Risk,
  3115.                                     rep.Odds );
  3116.                                  (void) rtEZRequestTags
  3117.                                     (foo,"OK",
  3118.                                     NULL,NULL,
  3119.                                     RT_Window,      map_window,
  3120.                                     RT_ReqPos,      REQPOS_CENTERSCR,
  3121.                                     RT_LockWindow,  TRUE,
  3122.                                     RTEZ_Flags,     EZREQF_CENTERTEXT,
  3123.                                     TAG_DONE );
  3124.                               } // End else no name
  3125.                            } // End if CanAttack or AtRisk
  3126.                          } // End for loop
  3127.                       } // End if response
  3128.                   } // End case 'z'
  3129.                   break;
  3130.                case 'c':
  3131.                   {  // center the display on the current unit icon
  3132.                      int ox = xoffs, oy = yoffs;
  3133.  
  3134.                      restore_hex_graphics(unit->col,unit->row,0);
  3135.                      set_display_offsets(unit->col,unit->row);
  3136.                      GP_smart_scroll(ox,oy);
  3137.                      if (blink_on)
  3138.                         plot_icon(unit->type,unit->col,unit->row,roster[unit->owner].color,ORDER_NONE,stacked);
  3139.                   }
  3140.                   break;
  3141.                case 'w': // Wait for unit (put at end of unit list
  3142.                   Remove((struct Node*)unit);
  3143.                   AddTail((struct List*)&unit_list,(struct Node*)unit);
  3144.                   /* And set the control flag so we don't try to do
  3145.                      anything else with this unit. */
  3146.                   restore_hex_graphics(unit->col,unit->row,0);
  3147.                   plot_icon(unit->type,unit->col,unit->row,
  3148.                   roster[unit->owner].color,ORDER_NONE,stacked);
  3149.                   control_flag = UNIT_LOST;
  3150.                   break;
  3151.                case ' ':   // skip this turn
  3152.                   // airplanes not moving still use fuel!
  3153.                   if( (wishbook[unit->type].range>0 ) && (
  3154.                            !(city_hereP(unit->col,unit->row))) &&
  3155.                            (unit->ship == NULL) )  { // added last factor trying to
  3156.                      // fix bug - when carrier moves with aircraft on board
  3157.                      // the aircraft lose fuel.
  3158.                      unit->fuel -= unit->move/60;
  3159.                      // Add a check for out of fuel
  3160.                      if (unit->fuel<=0) {
  3161.                         // aircraft out of fuel, crashes!
  3162.                         if (PLAYER.show & SHOW_REQ) {
  3163.                            plot_mapobject(unit->col,unit->row,154,30); // explosion
  3164.                            context_sound(BOOM_SOUND);              // kaboomy!
  3165.                            // now tell the player the awful news
  3166.                            sprintf(foo,"Your %s has run out of fuel and crashed!",
  3167.                               wishbook[unit->type].name);
  3168.                               (void)rtEZRequestTags(foo,"Oh no!",NULL,NULL,
  3169.                               RT_Window,     map_window,
  3170.                               RT_ReqPos,     REQPOS_CENTERWIN,
  3171.                               RT_LockWindow, TRUE,
  3172.                               TAG_DONE);
  3173.                         }
  3174.                         Remove((struct Node *)unit);           // kill the plane
  3175.                         roster[unit->owner].ulc[unit->type]++;
  3176.                         control_flag = UNIT_LOST;
  3177.                         destruct_unit(unit);
  3178.                         if (PLAYER.show & SHOW_GRP)  // Was SHOW_REQ ****
  3179.                            explore_at_hex(player,unit->col,unit->row,VISIBLE,FALSE);
  3180.                         else
  3181.                            explore_at_hex(player,unit->col,unit->row,INVISIBLE,FALSE);
  3182.                      } // End if fuel out
  3183.                   }
  3184.                   // damaged units in a city may be repaired
  3185.                   if (unit->damage>0 && unit->move>50 && city_hereP(unit->col ,unit->row))
  3186.                      unit->damage--;
  3187.                   unit->move = 0;
  3188.                   control_flag = UNIT_DONE;
  3189.                   break;
  3190.                case 'n':   // move on to next unit
  3191.                   // move unit to the tail end of the master unit list
  3192.                   Remove((struct Node *)unit);
  3193.                   AddTail((struct List *)&unit_list,(struct Node *)unit);
  3194.                   *current_unit = choose_default_unit(unit);
  3195.                   control_flag = UNIT_DONE;
  3196.                   break;
  3197.                case 'h':   /* head for home */
  3198.                   restore_hex_graphics(unit->col,unit->row,0);
  3199.                   plot_icon(unit->type,unit->col,unit->row,
  3200.                       roster[unit->owner].color,ORDER_GOTO,stacked);
  3201.                   give_orders(unit, ORDER_HOME, 0, 0, -1);
  3202.                   if( unit->orders)
  3203.                     unit->orders->processed = FALSE;
  3204.                   else
  3205.                       plot_icon(unit->type,unit->col,unit->row,
  3206.                           roster[unit->owner].color,ORDER_NONE,stacked);
  3207.                   control_flag = UNIT_DONE;
  3208.                   break;
  3209.                case 's':   /* unit for sentry duty */
  3210.                   // units that need fuel (i.e. aircraft) cannot sentry,
  3211.                   // except in cities or on aircraft carriers
  3212.                   if (wishbook[unit->type].range>0)
  3213.                      if (unit->ship==NULL && city_hereP(unit->col,unit->row)==NULL) {
  3214.                         playSound(DONK_SOUND,PLAYER.snd_vol);
  3215.                         break;   // we bail
  3216.                      FI
  3217.                   give_orders(unit,ORDER_SENTRY,0,0,-1);
  3218.                   explore_hex(player,unit->col,unit->row,VISIBLE,FALSE);
  3219.                   control_flag = UNIT_DONE;
  3220.                   break;
  3221.                case 'l':   // load ship
  3222.                   if (cargo_capacity(unit)<0) {
  3223.                      tell_user2("That unit is not a cargo vessel.",FALSE,DONK_SOUND);
  3224.                      break;
  3225.                   FI
  3226.                   if (unit->cargo>=cargo_capacity(unit) ||
  3227.                         unit->weight>=cargo_capacity(unit)) {
  3228.                      playSound(DONK_SOUND,PLAYER.snd_vol);
  3229.                      (void)rtEZRequestTags("That ship seems to be full,\nand cannot load more units.",
  3230.                         "Okay",NULL,NULL,
  3231.                         RT_Window,        map_window,
  3232.                         RT_ReqPos,        REQPOS_CENTERSCR,
  3233.                         RT_LockWindow,    TRUE,
  3234.                         RTEZ_Flags,       EZREQF_CENTERTEXT,
  3235.                         TAG_DONE );
  3236.                      break;
  3237.                   FI
  3238.                   give_orders(unit,ORDER_LOAD,0,0,0);
  3239.                   load_ship(unit);     // do it right away, if possible
  3240.                   start_blinking_unit(unit);    //***  so icon in buffer is
  3241.                   // updated ***/
  3242.                   if (unit->orders) {  // unit still not full
  3243.                      unit->orders->processed = TRUE;
  3244.                      control_flag = UNIT_DONE;
  3245.                   }                  break;
  3246.                case 'u':   // unload ship (clears orders of anything on board)
  3247.                   if (unit->cargo>0) {
  3248.                      struct Unit *cargo=(struct Unit *)unit_list.mlh_Head;
  3249.                      for (;cargo->unode.mln_Succ;cargo=(struct Unit *)cargo->unode.mln_Succ)
  3250.                         if (cargo->ship==unit)
  3251.                            clear_orders(cargo);
  3252.                   FI
  3253.                   break;
  3254.                case 'v':
  3255.                   control_flag = GO_SURVEY;
  3256.                   break;
  3257.                case '5':   // cycle or shuffle stack to next unit
  3258.                   {
  3259.                      struct Unit *newunit=shuffle_units(unit,FALSE);
  3260.  
  3261.                      if (newunit!=unit) {
  3262.                         *current_unit = newunit;
  3263.                         control_flag = UNIT_DONE;
  3264.                         break;
  3265.                      } else
  3266.                         if( PLAYER.soundfx == SOUND_ALL )
  3267.                            playSound(DONK_SOUND,PLAYER.snd_vol);
  3268.                   }
  3269.                   break;
  3270.                // following is a sort of movement table, translates
  3271.                // keypad numbers into movement directions
  3272.                case '6':
  3273.                   direction = EAST;
  3274.                   goto mm_moving;
  3275.                case '4':
  3276.                   direction = WEST;
  3277.                   goto mm_moving;
  3278.                case '7':
  3279.                   direction = NORTHWEST;
  3280.                   goto mm_moving;
  3281.                case '9':
  3282.                   direction = NORTHEAST;
  3283.                   goto mm_moving;
  3284.                case '1':
  3285.                   direction = SOUTHWEST;
  3286.                   goto mm_moving;
  3287.                case '3':
  3288.                   direction = SOUTHEAST;
  3289.                mm_moving:
  3290.                   {
  3291.                      int orgx=unit->col, orgy=unit->row;
  3292.                      // use orgx and orgy to track actual (not requested) movement
  3293.                      // of unit, so we can keep the blinking icon working right
  3294.                      restore_hex_graphics(unit->col,unit->row,0);
  3295.                      move_unit_dir(unit,direction);
  3296.                      // if the control flag is set for any reason, it means
  3297.                      // we are outta here; no need to reinstate blinking
  3298.                      if (control_flag)
  3299.                         break;
  3300.                      // if the unit has actually moved, we certainly
  3301.                      // need to reinstate the blinking icon
  3302.                      if (orgx!=unit->col || orgy!=unit->row) {
  3303.                         stacked = (count_units_at(unit->col,unit->row)>1);
  3304.                         start_blinking_unit(unit);
  3305.                      } else
  3306.                         if (blink_on)  /* didn't move: continue blinking normally */
  3307.                            plot_icon(unit->type,unit->col,unit->row,roster[unit->owner].color,ORDER_NONE,stacked);
  3308.                   }
  3309.                   break;
  3310.                default:
  3311.                   (void)rtEZRequestTags("That key has no defined function.\nPlease try one of the many other keys.",
  3312.                      "Okay",NULL,NULL,
  3313.                      RT_Window,        map_window,
  3314.                      RT_ReqPos,        REQPOS_CENTERSCR,
  3315.                      RT_LockWindow,    TRUE,
  3316.                      RTEZ_Flags,       EZREQF_CENTERTEXT,
  3317.                      TAG_DONE );
  3318.             } // end switch
  3319.          FI
  3320.          if (class==IDCMP_MOUSEBUTTONS && code==SELECTDOWN) {
  3321.             int x, y, ctr=0, orgx=unit->col, orgy=unit->row;
  3322.             int num_hexes=adjacent(unit->col,unit->row);
  3323.  
  3324.             abs_to_log(message->MouseX,message->MouseY,&x,&y);
  3325.             for (; ctr<num_hexes; ctr++)
  3326.                if (hexlist[ctr].col==x && hexlist[ctr].row==y) {
  3327.                   // use orgx and orgy to track actual (not requested) movement
  3328.                   // of unit, so we can keep the blinking icon working right
  3329.                   restore_hex_graphics(unit->col,unit->row,0);
  3330.                   move_unit_xy(unit,x,y);
  3331.                   // if the control flag is set for any reason, it means
  3332.                   // we are outta here; no need to reinstate blinking
  3333.                   if (control_flag)
  3334.                      break;
  3335.                   // if the unit has actually moved, we certainly
  3336.                   // need to reinstate the blinking icon
  3337.                   if (orgx!=unit->col || orgy!=unit->row) {
  3338.                      stacked = (count_units_at(unit->col,unit->row)>1);
  3339.                      start_blinking_unit(unit);
  3340.                   } else
  3341.                      if (blink_on)  /* didn't move: continue blinking normally */
  3342.                         plot_icon(unit->type,unit->col,unit->row,roster[unit->owner].color,ORDER_NONE,stacked);
  3343.                   break;   // no need for extra iterations after we found our hex
  3344.                FI
  3345.             //ROF
  3346.             if( ctr == num_hexes ) {
  3347.           /* There was not a match for the hexes immediately around
  3348.           **    the unit.  So, let's invoke a GOTO command for that
  3349.                **    unit. x and y contain the coordinates of the hex
  3350.                **    the user selected.
  3351.           */
  3352.              AI5_CalcPath( unit->type, unit->col, unit->row, x, y,
  3353.                 AI5_PATH_GOOD);
  3354.               if( Path[0] != -1 ) {
  3355.                 if( (wishbook[unit->type].range > 0) &&
  3356.                   (unit->fuel < PathLength) ) {
  3357.                   // We haven't got enough fuel to reach it
  3358.                   sprintf( foo,
  3359.                       "Can't reach from %ld,%ld to %ld,%ld, %s .",
  3360.                       unit->col, unit->row, x, y,
  3361.                       "\nNot enough fuel!");
  3362.                   (void)rtEZRequestTags
  3363.                       (foo,"Darn!",NULL,NULL,
  3364.                       RT_Window,        map_window,
  3365.                       RT_ReqPos,        REQPOS_CENTERSCR,
  3366.                        RT_LockWindow,    TRUE,
  3367.                    RTEZ_Flags,       EZREQF_CENTERTEXT,
  3368.                       TAG_DONE );
  3369.                   } // End if we haven't enough fuel
  3370.                 else {
  3371.                 int response;
  3372.                 int ETA = 1;
  3373.                   // We have a path, so verify the orders
  3374.                   // *** Add path drawing here ***
  3375.                 PathCost -= unit->move;
  3376.                  while (PathCost > 0) {
  3377.                      ETA++;
  3378.                      PathCost -= unit_speed(unit);
  3379.                  }
  3380.                   if( unit->name ) {
  3381.                       sprintf( foo,
  3382.                         "Move %s %s from %ld,%ld to %ld,%ld\nDistance is %ld\nPath length is %ld\nETA is %ld turns",
  3383.                           UnitString[unit->type],
  3384.                        unit->name, unit->col, unit->row, x, y,
  3385.                           AI5_GetDist(unit->col, unit->row, x, y),
  3386.                   PathLength,ETA);
  3387.                   }
  3388.                   else {
  3389.                       sprintf( foo,
  3390.                         "Move %s from %ld,%ld to %ld,%ld\nDistance is %ld\nPath length is %ld\nETA is %ld turns",
  3391.                           UnitString[unit->type],
  3392.                        unit->col, unit->row, x, y,
  3393.                           AI5_GetDist(unit->col, unit->row, x, y),
  3394.                       PathLength,ETA);
  3395.                   }
  3396.                  response = rtEZRequestTags
  3397.                       (foo,"OK|Patrol|Cancel",NULL,NULL,
  3398.                       RT_Window,        map_window,
  3399.                       RT_ReqPos,        REQPOS_CENTERSCR,
  3400.                       RT_LockWindow,    TRUE,
  3401.                       RTEZ_Flags,       EZREQF_CENTERTEXT,
  3402.                       TAG_DONE );
  3403.                  if (response == 1) {
  3404.                             restore_hex_graphics(unit->col,unit->row,0);
  3405.                             plot_icon(unit->type,unit->col,unit->row,
  3406.                                 roster[unit->owner].color,ORDER_GOTO,stacked);
  3407.                           give_orders(unit,ORDER_GOTO,x,y,PathLength);
  3408.                           unit->orders->processed = FALSE;
  3409.                           control_flag = UNIT_DONE;
  3410.                   }
  3411.                  if (response == 2) {
  3412.                      // Patrol looks the same except the order type
  3413.                         plot_icon(unit->type,unit->col,unit->row,
  3414.                             roster[unit->owner].color,ORDER_GOTO,stacked);
  3415.                       give_orders(unit,ORDER_PATROL,x,y,PathLength);
  3416.                       unit->orders->processed = FALSE;
  3417.                       control_flag = UNIT_DONE;
  3418.                   }
  3419.                   // *** Erase path drawn here ***
  3420.                 } // End else we have a path and enough fuel
  3421.               } // End if we have a path
  3422.               else {
  3423.                 // No path at all
  3424.                 sprintf( foo,
  3425.                   "Can't find a path from %ld,%ld to %ld,%ld .",
  3426.                   unit->col, unit->row, x, y);
  3427.                 if( wishbook[unit->type].range > 0 ) {
  3428.                    strcat( foo,
  3429.                       "\nOr path length exceeds range of aircraft." );
  3430.                 }
  3431.                 (void)rtEZRequestTags
  3432.                    (foo,"Darn!",NULL,NULL,
  3433.                    RT_Window,        map_window,
  3434.                    RT_ReqPos,        REQPOS_CENTERSCR,
  3435.                    RT_LockWindow,    TRUE,
  3436.                    RTEZ_Flags,       EZREQF_CENTERTEXT,
  3437.                    TAG_DONE );
  3438.               } // End else no path at all
  3439.                break;
  3440.             FI
  3441.          FI
  3442.          if (class==IDCMP_GADGETUP) {
  3443.             int ox = xoffs, oy = yoffs;
  3444.  
  3445.             restore_hex_graphics(unit->col,unit->row,0);
  3446.             if (scrolly(object,code)) {
  3447.                GP_smart_scroll(ox,oy);
  3448.             FI
  3449.             if (blink_on)
  3450.                plot_icon(unit->type,unit->col,unit->row,roster[unit->owner].color,ORDER_NONE,stacked);
  3451.          FI
  3452.          if (class == IDCMP_INTUITICKS)  // intuiticks control blinking
  3453.             if ((++ticks % 5) == 0) {
  3454.                blink_on = !blink_on;
  3455.                if (blink_on)
  3456.                   plot_icon(unit->type,unit->col,unit->row,roster[unit->owner].color,ORDER_NONE,stacked);
  3457.                else
  3458.                   restore_hex_graphics(unit->col,unit->row,0);
  3459.             FI
  3460.          ResetMenuStrip(map_window,move_menu_strip);
  3461.  
  3462.          switch (control_flag) {
  3463.             case UNIT_DONE:
  3464.             case UNIT_LOST:
  3465.             case GO_SURVEY:
  3466.             case GO_PRODUCTION:
  3467.                goto xyzzy;   // this should be "break 3;" but C makes it hard
  3468.          } // end switch
  3469.       OD
  3470.    OD
  3471.  xyzzy:
  3472.    // here is everything I need to clean up leaving movement mode
  3473.    if (control_flag==UNIT_LOST)
  3474.       unit=NULL;
  3475.    if (unit)
  3476.       GP_update_hex_display(unit->col,unit->row);
  3477.  
  3478.    // deactivate IDCMP event input
  3479.    ModifyIDCMP(map_window,NULL);
  3480.  
  3481.    // so that survey mode can pick up in the same spot
  3482.    if (unit) {
  3483.       cursx = unit->col;
  3484.       cursy = unit->row;
  3485.    FI
  3486.  
  3487.    // always clear the unit movement bar
  3488.    clear_movebar();
  3489.  
  3490.    /* return to my "generic" title bar, so no outdated information is */
  3491.    /* seen when it shouldn't be -- especially by the next player! */
  3492.    SetWindowTitles(map_window,"Game in progress...",(UBYTE *)~0);
  3493. }
  3494.  
  3495.  
  3496. void plot_cursor(x,y)
  3497. int x,y;
  3498. {
  3499.    int px, py;
  3500.  
  3501.    // plot the cursor; grafx_bitmap coords 242,30
  3502.    log_to_abs(x,y,&px,&py);
  3503.    px += 5;   py += 8;  // centering it in the hexagon
  3504.    BltBitMapRastPort(&grafx_bitmap,242,30,rast_port,px,py,21,17,0x0C0);
  3505. }
  3506.  
  3507.  
  3508. // move cursor in the specified direction, if possible
  3509. void move_cursor_dir(dir)
  3510. int dir;
  3511. {
  3512.    int targx = cursx, targy = cursy;
  3513.  
  3514.    switch (dir) {
  3515.       case EAST:
  3516.          targx++;
  3517.          break;
  3518.       case WEST:
  3519.          targx--;
  3520.          break;
  3521.       case NORTHEAST:
  3522.          targy--;
  3523.          if (cursy%2) // odd number
  3524.             targx++;
  3525.          break;
  3526.       case NORTHWEST:
  3527.          targy--;
  3528.          if (!(cursy%2)) // even number
  3529.             targx--;
  3530.          break;
  3531.       case SOUTHEAST:
  3532.          targy++;
  3533.          if (cursy%2) // odd number
  3534.             targx++;
  3535.          break;
  3536.       case SOUTHWEST:
  3537.          targy++;
  3538.          if (!(cursy%2)) // even number
  3539.             targx--;
  3540.          break;
  3541.       default:
  3542.          return;
  3543.    }
  3544.  
  3545.    /* correct values for wrap, if it's active */
  3546.    if (wrap) {
  3547.       if (targx<0)
  3548.          targx += width;
  3549.       if (targx>width)
  3550.          targx -= width;
  3551.       if (targy<0)
  3552.          targy += height;
  3553.       if (targy>height)
  3554.          targy -= height;
  3555.    }
  3556.  
  3557.    if (targx<0 || targx>(width-1))
  3558.       return;
  3559.    if (targy<0 || targy>(height-1))
  3560.       return;
  3561.  
  3562.    cursx = targx;
  3563.    cursy = targy;
  3564. }
  3565.  
  3566.  
  3567. // end of listing
  3568.  
  3569.  
  3570.